UNPKG

gentics-ui-core

Version:

This is the common core framework for the Gentics CMS and Mesh UI, and other Angular applications.

139 lines 21.4 kB
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, ContentChildren, ElementRef, Input, QueryList } from '@angular/core'; import { coerceToBoolean } from '../../common/coerce-to-boolean'; import { DropdownItem } from '../dropdown-list/dropdown-item.component'; import { SplitButtonPrimaryAction } from './split-button-primary-action.component'; import * as i0 from "@angular/core"; import * as i1 from "../button/button.component"; import * as i2 from "../dropdown-list/dropdown-list.component"; import * as i3 from "../dropdown-list/dropdown-content.component"; import * as i4 from "../../directives/autofocus/autofocus.directive"; import * as i5 from "@angular/common"; import * as i6 from "../dropdown-list/dropdown-trigger.directive"; import * as i7 from "../icon/icon.directive"; /** * A split button component. * * The main content of the button and the handler of the main click event is specified by * using a `<gtx-split-button-primary-action>` child element. * * Secondary actions can be defined using `<gtx-dropdown-item>` child elements and * their click handlers. If secondary actions are defined, a dropdown trigger * will be displayed to the right of the main content. * * All input properties of `<gtx-button>`, except for `icon` and `submit` are supported. * * This component depends on the `<gtx-overlay-host>` being present in the app. * * ```html * <gtx-split-button> * <gtx-split-button-primary-action (click)="save()">Save Document</gtx-split-button-primary-action> * <gtx-dropdown-item (click)="saveAndPublish()">Save and Publish</gtx-dropdown-item> * <gtx-dropdown-item (click)="saveAndEmail()">Save and Send via E-Mail</gtx-dropdown-item> * </gtx-split-button> * ``` */ export class SplitButton { constructor(changeDetector) { this.changeDetector = changeDetector; /** * Sets the input field to be auto-focused. Handled by `AutofocusDirective`. */ this.autofocus = false; /** * Specify the size of the button. Can be "small", "regular" or "large". */ this.size = 'regular'; /** * Type determines the style of the button. Can be "default", "secondary", * "success", "warning" or "alert". */ this.type = 'default'; this.isFlat = false; this.isDisabled = false; } /** * Setting the "flat" attribute gives the button a transparent background * and only depth on hover. */ get flat() { return this.isFlat; } set flat(value) { this.isFlat = coerceToBoolean(value); } /** * Controls whether the button is disabled. */ get disabled() { return this.isDisabled; } set disabled(value) { this.isDisabled = coerceToBoolean(value); } ngAfterViewInit() { this.queryListSub = this.secondaryActions.changes.subscribe(() => this.changeDetector.markForCheck()); this.secondaryActions.notifyOnChanges(); } ngOnDestroy() { if (this.queryListSub) { this.queryListSub.unsubscribe(); this.queryListSub = null; } } /** * If the user does not click the primary action content itself, but the button around it, * this method is used to trigger a click event on the primary action content. */ onPrimaryButtonClick(event) { if (this.primaryAction && this.isFromPrimaryButton(event)) { this.primaryAction.nativeElement.click(); } } /** * Returns true if the event was fired by the primary `gtx-button` element or one of its children, * but not by the `gtx-split-button-primary-action` or one of its children. */ isFromPrimaryButton(event) { if (event && event.srcElement && event.srcElement instanceof Node) { // We traverse the DOM tree upwards from the srcElement. // If we first find a gtx-split-button-primary-action, the click came from // inside the primary action content. // If we find a gtx-button first, the click didn't come from inside the primary action content. let currNode = event.srcElement; do { switch (currNode.nodeName) { case 'GTX-BUTTON': return true; case 'GTX-SPLIT-BUTTON-PRIMARY-ACTION': return false; default: break; } } while (currNode = currNode.parentNode); } return false; } } /** @nocollapse */ SplitButton.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: SplitButton, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ SplitButton.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: SplitButton, selector: "gtx-split-button", inputs: { autofocus: "autofocus", size: "size", type: "type", flat: "flat", disabled: "disabled" }, queries: [{ propertyName: "primaryAction", first: true, predicate: SplitButtonPrimaryAction, descendants: true, read: ElementRef, static: true }, { propertyName: "secondaryActions", predicate: DropdownItem }], ngImport: i0, template: "<div class=\"split-button-wrapper z-depth-1\">\n\n <gtx-button class=\"primary-button\"\n [autofocus]=\"autofocus\"\n [size]=\"size\"\n [type]=\"type\"\n [flat]=\"isFlat\"\n [disabled]=\"isDisabled\"\n [class.has-secondary-actions]=\"secondaryActions.length > 0\"\n (click)=\"onPrimaryButtonClick($event)\">\n <ng-content select=\"gtx-split-button-primary-action\"></ng-content>\n </gtx-button>\n\n <ng-container *ngIf=\"secondaryActions.length > 0\">\n\n <div class=\"spacer-line\"></div>\n\n <gtx-dropdown-list align=\"right\" belowTrigger=\"true\">\n <gtx-dropdown-trigger>\n <gtx-button class=\"more-trigger\"\n [size]=\"size\"\n [type]=\"type\"\n [flat]=\"isFlat\"\n [disabled]=\"isDisabled\">\n <icon>arrow_drop_down</icon>\n </gtx-button>\n </gtx-dropdown-trigger>\n <gtx-dropdown-content>\n <ng-content select=\"gtx-dropdown-item\"></ng-content>\n </gtx-dropdown-content>\n </gtx-dropdown-list>\n\n </ng-container>\n\n</div>\n", components: [{ type: i1.Button, selector: "gtx-button", inputs: ["autofocus", "size", "type", "flat", "icon", "disabled", "submit"] }, { type: i2.DropdownList, selector: "gtx-dropdown-list", inputs: ["align", "width", "belowTrigger", "sticky", "closeOnEscape", "disabled"], outputs: ["open", "close"] }, { type: i3.DropdownContent, selector: "gtx-dropdown-content" }], directives: [{ type: i4.AutofocusDirective, selector: "\n gtx-button[autofocus],\n gtx-checkbox[autofocus],\n gtx-date-time-picker[autofocus],\n gtx-file-picker[autofocus],\n gtx-input[autofocus],\n gtx-radio-button[autofocus],\n gtx-search-bar[autofocus],\n gtx-select[autofocus],\n gtx-textarea[autofocus]", inputs: ["autofocus"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.DropdownTriggerDirective, selector: "gtx-dropdown-trigger" }, { type: i7.Icon, selector: "icon" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: SplitButton, decorators: [{ type: Component, args: [{ selector: 'gtx-split-button', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"split-button-wrapper z-depth-1\">\n\n <gtx-button class=\"primary-button\"\n [autofocus]=\"autofocus\"\n [size]=\"size\"\n [type]=\"type\"\n [flat]=\"isFlat\"\n [disabled]=\"isDisabled\"\n [class.has-secondary-actions]=\"secondaryActions.length > 0\"\n (click)=\"onPrimaryButtonClick($event)\">\n <ng-content select=\"gtx-split-button-primary-action\"></ng-content>\n </gtx-button>\n\n <ng-container *ngIf=\"secondaryActions.length > 0\">\n\n <div class=\"spacer-line\"></div>\n\n <gtx-dropdown-list align=\"right\" belowTrigger=\"true\">\n <gtx-dropdown-trigger>\n <gtx-button class=\"more-trigger\"\n [size]=\"size\"\n [type]=\"type\"\n [flat]=\"isFlat\"\n [disabled]=\"isDisabled\">\n <icon>arrow_drop_down</icon>\n </gtx-button>\n </gtx-dropdown-trigger>\n <gtx-dropdown-content>\n <ng-content select=\"gtx-dropdown-item\"></ng-content>\n </gtx-dropdown-content>\n </gtx-dropdown-list>\n\n </ng-container>\n\n</div>\n" }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { autofocus: [{ type: Input }], size: [{ type: Input }], type: [{ type: Input }], flat: [{ type: Input }], disabled: [{ type: Input }], primaryAction: [{ type: ContentChild, args: [SplitButtonPrimaryAction, { read: ElementRef, static: true }] }], secondaryActions: [{ type: ContentChildren, args: [DropdownItem] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"split-button.component.js","sourceRoot":"","sources":["../../../../../src/components/split-button/split-button.component.ts","../../../../../src/components/split-button/split-button.tpl.html"],"names":[],"mappings":"AAAA,OAAO,EAEH,uBAAuB,EACvB,iBAAiB,EACjB,SAAS,EACT,YAAY,EACZ,eAAe,EACf,UAAU,EACV,KAAK,EAEL,SAAS,EACZ,MAAM,eAAe,CAAC;AAGvB,OAAO,EAAC,eAAe,EAAC,MAAM,gCAAgC,CAAC;AAC/D,OAAO,EAAC,YAAY,EAAC,MAAM,0CAA0C,CAAC;AACtE,OAAO,EAAC,wBAAwB,EAAC,MAAM,yCAAyC,CAAC;;;;;;;;;AAEjF;;;;;;;;;;;;;;;;;;;;;GAqBG;AAMH,MAAM,OAAO,WAAW;IAoDpB,YAAoB,cAAiC;QAAjC,mBAAc,GAAd,cAAc,CAAmB;QAlDrD;;WAEG;QACM,cAAS,GAAY,KAAK,CAAC;QAEpC;;WAEG;QACM,SAAI,GAAkC,SAAS,CAAC;QAEzD;;;WAGG;QACM,SAAI,GAA8D,SAAS,CAAC;QA+BrF,WAAM,GAAG,KAAK,CAAC;QACf,eAAU,GAAG,KAAK,CAAC;IAIqC,CAAC;IAlCzD;;;OAGG;IACH,IACI,IAAI;QACJ,OAAO,IAAI,CAAC,MAAM,CAAC;IACvB,CAAC;IACD,IAAI,IAAI,CAAC,KAAc;QACnB,IAAI,CAAC,MAAM,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC;IAED;;OAEG;IACH,IACI,QAAQ;QACR,OAAO,IAAI,CAAC,UAAU,CAAC;IAC3B,CAAC;IACD,IAAI,QAAQ,CAAC,KAAc;QACvB,IAAI,CAAC,UAAU,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;IAC7C,CAAC;IAeD,eAAe;QACX,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,SAAS,CACvD,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,CAC3C,CAAC;QACF,IAAI,CAAC,gBAAgB,CAAC,eAAe,EAAE,CAAC;IAC5C,CAAC;IAED,WAAW;QACP,IAAI,IAAI,CAAC,YAAY,EAAE;YACnB,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;YAChC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;SAC5B;IACL,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,KAAY;QAC7B,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,EAAE;YACvD,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;SAC5C;IACL,CAAC;IAED;;;OAGG;IACK,mBAAmB,CAAC,KAAY;QACpC,IAAI,KAAK,IAAI,KAAK,CAAC,UAAU,IAAI,KAAK,CAAC,UAAU,YAAY,IAAI,EAAE;YAC/D,wDAAwD;YACxD,0EAA0E;YAC1E,qCAAqC;YACrC,+FAA+F;YAC/F,IAAI,QAAQ,GAAS,KAAK,CAAC,UAAU,CAAC;YACtC,GAAG;gBACC,QAAQ,QAAQ,CAAC,QAAQ,EAAE;oBACvB,KAAK,YAAY;wBACb,OAAO,IAAI,CAAC;oBAChB,KAAK,iCAAiC;wBAClC,OAAO,KAAK,CAAC;oBACjB;wBACI,MAAM;iBACb;aACJ,QAAQ,QAAQ,GAAG,QAAQ,CAAC,UAAU,EAAE;SAC5C;QACD,OAAO,KAAK,CAAC;IACjB,CAAC;;2HArGQ,WAAW;+GAAX,WAAW,uMAyCN,wBAAwB,2BAAU,UAAU,iEAGzC,YAAY,6BCzFjC,quCAmCA;2FDUa,WAAW;kBALvB,SAAS;+BACI,kBAAkB,mBAEX,uBAAuB,CAAC,MAAM;wGAOtC,SAAS;sBAAjB,KAAK;gBAKG,IAAI;sBAAZ,KAAK;gBAMG,IAAI;sBAAZ,KAAK;gBAOF,IAAI;sBADP,KAAK;gBAYF,QAAQ;sBADX,KAAK;gBASN,aAAa;sBADZ,YAAY;uBAAC,wBAAwB,EAAE,EAAE,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,IAAI,EAAE;gBAI1E,gBAAgB;sBADf,eAAe;uBAAC,YAAY","sourcesContent":["import {\n    AfterViewInit,\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChild,\n    ContentChildren,\n    ElementRef,\n    Input,\n    OnDestroy,\n    QueryList\n} from '@angular/core';\nimport {Subscription} from 'rxjs';\n\nimport {coerceToBoolean} from '../../common/coerce-to-boolean';\nimport {DropdownItem} from '../dropdown-list/dropdown-item.component';\nimport {SplitButtonPrimaryAction} from './split-button-primary-action.component';\n\n/**\n * A split button component.\n *\n * The main content of the button and the handler of the main click event is specified by\n * using a `<gtx-split-button-primary-action>` child element.\n *\n * Secondary actions can be defined using `<gtx-dropdown-item>` child elements and\n * their click handlers. If secondary actions are defined, a dropdown trigger\n * will be displayed to the right of the main content.\n *\n * All input properties of `<gtx-button>`, except for `icon` and `submit` are supported.\n *\n * This component depends on the `<gtx-overlay-host>` being present in the app.\n *\n * ```html\n * <gtx-split-button>\n *     <gtx-split-button-primary-action (click)=\"save()\">Save Document</gtx-split-button-primary-action>\n *     <gtx-dropdown-item (click)=\"saveAndPublish()\">Save and Publish</gtx-dropdown-item>\n *     <gtx-dropdown-item (click)=\"saveAndEmail()\">Save and Send via E-Mail</gtx-dropdown-item>\n * </gtx-split-button>\n * ```\n */\n@Component({\n    selector: 'gtx-split-button',\n    templateUrl: './split-button.tpl.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class SplitButton implements AfterViewInit, OnDestroy {\n\n    /**\n     * Sets the input field to be auto-focused. Handled by `AutofocusDirective`.\n     */\n    @Input() autofocus: boolean = false;\n\n    /**\n     * Specify the size of the button. Can be \"small\", \"regular\" or \"large\".\n     */\n    @Input() size: 'small' | 'regular' | 'large' = 'regular';\n\n    /**\n     * Type determines the style of the button. Can be \"default\", \"secondary\",\n     * \"success\", \"warning\" or \"alert\".\n     */\n    @Input() type: 'default' | 'secondary' | 'success' | 'warning' | 'alert' = 'default';\n\n    /**\n     * Setting the \"flat\" attribute gives the button a transparent background\n     * and only depth on hover.\n     */\n    @Input()\n    get flat(): boolean {\n        return this.isFlat;\n    }\n    set flat(value: boolean) {\n        this.isFlat = coerceToBoolean(value);\n    }\n\n    /**\n     * Controls whether the button is disabled.\n     */\n    @Input()\n    get disabled(): boolean {\n        return this.isDisabled;\n    }\n    set disabled(value: boolean) {\n        this.isDisabled = coerceToBoolean(value);\n    }\n\n    @ContentChild(SplitButtonPrimaryAction, { read: ElementRef, static: true })\n    primaryAction: ElementRef;\n\n    @ContentChildren(DropdownItem)\n    secondaryActions: QueryList<DropdownItem>;\n\n    isFlat = false;\n    isDisabled = false;\n\n    private queryListSub: Subscription;\n\n    constructor(private changeDetector: ChangeDetectorRef) {}\n\n    ngAfterViewInit(): void {\n        this.queryListSub = this.secondaryActions.changes.subscribe(\n            () => this.changeDetector.markForCheck()\n        );\n        this.secondaryActions.notifyOnChanges();\n    }\n\n    ngOnDestroy(): void {\n        if (this.queryListSub) {\n            this.queryListSub.unsubscribe();\n            this.queryListSub = null;\n        }\n    }\n\n    /**\n     * If the user does not click the primary action content itself, but the button around it,\n     * this method is used to trigger a click event on the primary action content.\n     */\n    onPrimaryButtonClick(event: Event): void {\n        if (this.primaryAction && this.isFromPrimaryButton(event)) {\n            this.primaryAction.nativeElement.click();\n        }\n    }\n\n    /**\n     * Returns true if the event was fired by the primary `gtx-button` element or one of its children,\n     * but not by the `gtx-split-button-primary-action` or one of its children.\n     */\n    private isFromPrimaryButton(event: Event): boolean {\n        if (event && event.srcElement && event.srcElement instanceof Node) {\n            // We traverse the DOM tree upwards from the srcElement.\n            // If we first find a gtx-split-button-primary-action, the click came from\n            // inside the primary action content.\n            // If we find a gtx-button first, the click didn't come from inside the primary action content.\n            let currNode: Node = event.srcElement;\n            do {\n                switch (currNode.nodeName) {\n                    case 'GTX-BUTTON':\n                        return true;\n                    case 'GTX-SPLIT-BUTTON-PRIMARY-ACTION':\n                        return false;\n                    default:\n                        break;\n                }\n            } while (currNode = currNode.parentNode);\n        }\n        return false;\n    }\n\n}\n","<div class=\"split-button-wrapper z-depth-1\">\n\n    <gtx-button class=\"primary-button\"\n            [autofocus]=\"autofocus\"\n            [size]=\"size\"\n            [type]=\"type\"\n            [flat]=\"isFlat\"\n            [disabled]=\"isDisabled\"\n            [class.has-secondary-actions]=\"secondaryActions.length > 0\"\n            (click)=\"onPrimaryButtonClick($event)\">\n        <ng-content select=\"gtx-split-button-primary-action\"></ng-content>\n    </gtx-button>\n\n    <ng-container *ngIf=\"secondaryActions.length > 0\">\n\n        <div class=\"spacer-line\"></div>\n\n        <gtx-dropdown-list align=\"right\" belowTrigger=\"true\">\n            <gtx-dropdown-trigger>\n                <gtx-button class=\"more-trigger\"\n                        [size]=\"size\"\n                        [type]=\"type\"\n                        [flat]=\"isFlat\"\n                        [disabled]=\"isDisabled\">\n                    <icon>arrow_drop_down</icon>\n                </gtx-button>\n            </gtx-dropdown-trigger>\n            <gtx-dropdown-content>\n                <ng-content select=\"gtx-dropdown-item\"></ng-content>\n            </gtx-dropdown-content>\n        </gtx-dropdown-list>\n\n    </ng-container>\n\n</div>\n"]}