UNPKG

@clr/angular

Version:

Angular components for Clarity

236 lines (234 loc) 23.1 kB
/* * Copyright (c) 2016-2025 Broadcom. All Rights Reserved. * The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. * This software is released under MIT license. * The full license information can be found in LICENSE in the root directory of this project. */ import { Component, ContentChild, EventEmitter, HostBinding, Input, Optional, Output, SkipSelf, } from '@angular/core'; import { uniqueIdFactory } from '../../utils/id-generator/id-generator.service'; import { ClrStackViewLabel } from './stack-view-custom-tags'; import * as i0 from "@angular/core"; import * as i1 from "../../utils/i18n/common-strings.service"; import * as i2 from "@angular/common"; import * as i3 from "../../icon/icon"; import * as i4 from "../../utils/animations/expandable-animation/expandable-animation"; export class ClrStackBlock { /* * This would be more efficient with @ContentChildren, with the parent ClrStackBlock * querying for children StackBlocks, but this feature is not available when downgrading * the component for Angular 1. */ constructor(parent, commonStrings) { this.parent = parent; this.commonStrings = commonStrings; this.expanded = false; this.expandable = false; this.expandedChange = new EventEmitter(false); this.focused = false; this.uniqueId = uniqueIdFactory(); this._changedChildren = 0; this._fullyInitialized = false; this._changed = false; if (parent) { parent.addChild(); } } set setChangedValue(value) { this._changed = value; if (this.parent && this._fullyInitialized) { if (value) { this.parent._changedChildren++; } else { this.parent._changedChildren--; } } } get getChangedValue() { return this._changed || (this._changedChildren > 0 && !this.expanded); } get onStackLabelFocus() { return this.expandable && !this.expanded && this.focused; } get labelledById() { return this.stackBlockTitle.id; } get headingLevel() { if (this.ariaLevel) { return this.ariaLevel + ''; } return this.parent ? '4' : '3'; } get caretDirection() { return this.expanded ? 'down' : 'right'; } get role() { return this.expandable ? 'button' : null; } get tabIndex() { return this.expandable ? '0' : null; } get ariaExpanded() { if (!this.expandable) { return null; } else { return this.expanded ? 'true' : 'false'; } } ngOnInit() { // in order to access the parent ClrStackBlock's properties, // the child ClrStackBlock has to be fully initialized at first. this._fullyInitialized = true; } addChild() { this.expandable = true; } toggleExpand(event) { if (eventIsInputEvent(event)) { return; } if (this.expandable) { this.expanded = !this.expanded; this.expandedChange.emit(this.expanded); } } getStackChildrenId() { return this.expanded ? `clr-stack-children-${this.uniqueId}` : null; } preventDefaultIfNotInputEvent(event) { if (eventIsInputEvent(event)) { return; } event.preventDefault(); } } ClrStackBlock.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrStackBlock, deps: [{ token: ClrStackBlock, optional: true, skipSelf: true }, { token: i1.ClrCommonStringsService }], target: i0.ɵɵFactoryTarget.Component }); ClrStackBlock.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.2", type: ClrStackBlock, selector: "clr-stack-block", inputs: { expanded: ["clrSbExpanded", "expanded"], expandable: ["clrSbExpandable", "expandable"], ariaLevel: ["clrStackViewLevel", "ariaLevel"], setChangedValue: ["clrSbNotifyChange", "setChangedValue"] }, outputs: { expandedChange: "clrSbExpandedChange" }, host: { properties: { "class.stack-block": "true", "attr.role": "\"heading\"", "attr.aria-level": "headingLevel", "class.stack-block-expanded": "this.expanded", "class.stack-block-expandable": "this.expandable", "class.stack-block-changed": "this.getChangedValue", "class.on-focus": "this.onStackLabelFocus" } }, queries: [{ propertyName: "stackBlockTitle", first: true, predicate: ClrStackViewLabel, descendants: true }], ngImport: i0, template: ` <!-- The 'preventDefault' for the space keydown event prevents the page from scrolling when a stack block is toggled via the space key. --> <div class="stack-block-label" (click)="toggleExpand($event)" (keyup.enter)="toggleExpand($event)" (keyup.space)="toggleExpand($event)" (keydown.space)="preventDefaultIfNotInputEvent($event)" (focus)="focused = true" (blur)="focused = false" [id]="uniqueId" [attr.role]="role" [attr.tabindex]="tabIndex" [attr.aria-expanded]="ariaExpanded" [attr.aria-controls]="getStackChildrenId()" > <cds-icon shape="angle" class="stack-block-caret" *ngIf="expandable" [attr.direction]="caretDirection"></cds-icon> <span class="clr-sr-only" *ngIf="getChangedValue">{{ commonStrings.keys.stackViewChanged }}</span> <div class="stack-view-key"> <!-- This structure changed to fix #3567 and the a11y request was to move away from dl's --> <!-- I added the key class to update css targets for the original component style --> <ng-content select="clr-stack-label"></ng-content> </div> <div class="stack-block-content"> <ng-content></ng-content> </div> </div> <clr-expandable-animation [clrExpandTrigger]="expanded" class="stack-children"> <div [style.height]="expanded ? 'auto' : 0" role="region" *ngIf="expanded" [attr.id]="getStackChildrenId()" [attr.aria-labelledby]="labelledById" > <ng-content select="clr-stack-block"></ng-content> </div> </clr-expandable-animation> `, isInline: true, styles: [":host{display:block}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.CdsIconCustomTag, selector: "cds-icon" }, { kind: "component", type: i4.ClrExpandableAnimation, selector: "clr-expandable-animation", inputs: ["clrExpandTrigger"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrStackBlock, decorators: [{ type: Component, args: [{ selector: 'clr-stack-block', template: ` <!-- The 'preventDefault' for the space keydown event prevents the page from scrolling when a stack block is toggled via the space key. --> <div class="stack-block-label" (click)="toggleExpand($event)" (keyup.enter)="toggleExpand($event)" (keyup.space)="toggleExpand($event)" (keydown.space)="preventDefaultIfNotInputEvent($event)" (focus)="focused = true" (blur)="focused = false" [id]="uniqueId" [attr.role]="role" [attr.tabindex]="tabIndex" [attr.aria-expanded]="ariaExpanded" [attr.aria-controls]="getStackChildrenId()" > <cds-icon shape="angle" class="stack-block-caret" *ngIf="expandable" [attr.direction]="caretDirection"></cds-icon> <span class="clr-sr-only" *ngIf="getChangedValue">{{ commonStrings.keys.stackViewChanged }}</span> <div class="stack-view-key"> <!-- This structure changed to fix #3567 and the a11y request was to move away from dl's --> <!-- I added the key class to update css targets for the original component style --> <ng-content select="clr-stack-label"></ng-content> </div> <div class="stack-block-content"> <ng-content></ng-content> </div> </div> <clr-expandable-animation [clrExpandTrigger]="expanded" class="stack-children"> <div [style.height]="expanded ? 'auto' : 0" role="region" *ngIf="expanded" [attr.id]="getStackChildrenId()" [attr.aria-labelledby]="labelledById" > <ng-content select="clr-stack-block"></ng-content> </div> </clr-expandable-animation> `, host: { '[class.stack-block]': 'true', '[attr.role]': '"heading"', '[attr.aria-level]': 'headingLevel', }, styles: [":host{display:block}\n"] }] }], ctorParameters: function () { return [{ type: ClrStackBlock, decorators: [{ type: SkipSelf }, { type: Optional }] }, { type: i1.ClrCommonStringsService }]; }, propDecorators: { expanded: [{ type: Input, args: ['clrSbExpanded'] }, { type: HostBinding, args: ['class.stack-block-expanded'] }], expandable: [{ type: Input, args: ['clrSbExpandable'] }, { type: HostBinding, args: ['class.stack-block-expandable'] }], ariaLevel: [{ type: Input, args: ['clrStackViewLevel'] }], expandedChange: [{ type: Output, args: ['clrSbExpandedChange'] }], stackBlockTitle: [{ type: ContentChild, args: [ClrStackViewLabel] }], setChangedValue: [{ type: Input, args: ['clrSbNotifyChange'] }], getChangedValue: [{ type: HostBinding, args: ['class.stack-block-changed'] }], onStackLabelFocus: [{ type: HostBinding, args: ['class.on-focus'] }] } }); function eventIsInputEvent(event) { const targetElement = event?.target; return targetElement?.tagName ? ['INPUT', 'TEXTAREA', 'BUTTON', 'A', 'SELECT', 'OPTION'].includes(targetElement.tagName) : false; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-block.js","sourceRoot":"","sources":["../../../../../projects/angular/src/data/stack-view/stack-block.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2016-2025 Broadcom. All Rights Reserved.\n * The term \"Broadcom\" refers to Broadcom Inc. and/or its subsidiaries.\n * This software is released under MIT license.\n * The full license information can be found in LICENSE in the root directory of this project.\n */\n\nimport {\n  Component,\n  ContentChild,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnInit,\n  Optional,\n  Output,\n  SkipSelf,\n} from '@angular/core';\n\nimport { ClrCommonStringsService } from '../../utils/i18n/common-strings.service';\nimport { uniqueIdFactory } from '../../utils/id-generator/id-generator.service';\nimport { ClrStackViewLabel } from './stack-view-custom-tags';\n\n@Component({\n  selector: 'clr-stack-block',\n  template: `\n    <!-- The 'preventDefault' for the space keydown event prevents the page\n         from scrolling when a stack block is toggled via the space key. -->\n    <div\n      class=\"stack-block-label\"\n      (click)=\"toggleExpand($event)\"\n      (keyup.enter)=\"toggleExpand($event)\"\n      (keyup.space)=\"toggleExpand($event)\"\n      (keydown.space)=\"preventDefaultIfNotInputEvent($event)\"\n      (focus)=\"focused = true\"\n      (blur)=\"focused = false\"\n      [id]=\"uniqueId\"\n      [attr.role]=\"role\"\n      [attr.tabindex]=\"tabIndex\"\n      [attr.aria-expanded]=\"ariaExpanded\"\n      [attr.aria-controls]=\"getStackChildrenId()\"\n    >\n      <cds-icon shape=\"angle\" class=\"stack-block-caret\" *ngIf=\"expandable\" [attr.direction]=\"caretDirection\"></cds-icon>\n      <span class=\"clr-sr-only\" *ngIf=\"getChangedValue\">{{ commonStrings.keys.stackViewChanged }}</span>\n      <div class=\"stack-view-key\">\n        <!-- This structure changed to fix #3567 and the a11y request was to move away from dl's -->\n        <!-- I added the key class to update css targets for the original component style -->\n        <ng-content select=\"clr-stack-label\"></ng-content>\n      </div>\n      <div class=\"stack-block-content\">\n        <ng-content></ng-content>\n      </div>\n    </div>\n\n    <clr-expandable-animation [clrExpandTrigger]=\"expanded\" class=\"stack-children\">\n      <div\n        [style.height]=\"expanded ? 'auto' : 0\"\n        role=\"region\"\n        *ngIf=\"expanded\"\n        [attr.id]=\"getStackChildrenId()\"\n        [attr.aria-labelledby]=\"labelledById\"\n      >\n        <ng-content select=\"clr-stack-block\"></ng-content>\n      </div>\n    </clr-expandable-animation>\n  `,\n  // Custom elements are inline by default\n  styles: [\n    `\n      :host {\n        display: block;\n      }\n    `,\n  ],\n  // Make sure the host has the proper class for styling purposes\n  host: {\n    '[class.stack-block]': 'true',\n    '[attr.role]': '\"heading\"',\n    '[attr.aria-level]': 'headingLevel',\n  },\n})\nexport class ClrStackBlock implements OnInit {\n  @Input('clrSbExpanded') @HostBinding('class.stack-block-expanded') expanded = false;\n  @Input('clrSbExpandable') @HostBinding('class.stack-block-expandable') expandable = false;\n\n  /**\n   * Depth of the stack view starting from 1 for first level\n   */\n  @Input('clrStackViewLevel') ariaLevel: number;\n\n  @Output('clrSbExpandedChange') expandedChange = new EventEmitter<boolean>(false);\n\n  @ContentChild(ClrStackViewLabel) stackBlockTitle: any;\n\n  focused = false;\n  uniqueId = uniqueIdFactory();\n\n  private _changedChildren = 0;\n  private _fullyInitialized = false;\n  private _changed = false;\n\n  /*\n   * This would be more efficient with @ContentChildren, with the parent ClrStackBlock\n   * querying for children StackBlocks, but this feature is not available when downgrading\n   * the component for Angular 1.\n   */\n  constructor(\n    @SkipSelf()\n    @Optional()\n    private parent: ClrStackBlock,\n    public commonStrings: ClrCommonStringsService\n  ) {\n    if (parent) {\n      parent.addChild();\n    }\n  }\n\n  @Input('clrSbNotifyChange')\n  set setChangedValue(value: boolean) {\n    this._changed = value;\n\n    if (this.parent && this._fullyInitialized) {\n      if (value) {\n        this.parent._changedChildren++;\n      } else {\n        this.parent._changedChildren--;\n      }\n    }\n  }\n\n  @HostBinding('class.stack-block-changed')\n  get getChangedValue(): boolean {\n    return this._changed || (this._changedChildren > 0 && !this.expanded);\n  }\n\n  @HostBinding('class.on-focus')\n  get onStackLabelFocus(): boolean {\n    return this.expandable && !this.expanded && this.focused;\n  }\n\n  get labelledById() {\n    return this.stackBlockTitle.id;\n  }\n\n  get headingLevel() {\n    if (this.ariaLevel) {\n      return this.ariaLevel + '';\n    }\n\n    return this.parent ? '4' : '3';\n  }\n\n  get caretDirection(): string {\n    return this.expanded ? 'down' : 'right';\n  }\n\n  get role(): string {\n    return this.expandable ? 'button' : null;\n  }\n\n  get tabIndex(): string {\n    return this.expandable ? '0' : null;\n  }\n\n  get ariaExpanded(): string {\n    if (!this.expandable) {\n      return null;\n    } else {\n      return this.expanded ? 'true' : 'false';\n    }\n  }\n\n  ngOnInit(): void {\n    // in order to access the parent ClrStackBlock's properties,\n    // the child ClrStackBlock has to be fully initialized at first.\n    this._fullyInitialized = true;\n  }\n\n  addChild(): void {\n    this.expandable = true;\n  }\n\n  toggleExpand(event?: Event): void {\n    if (eventIsInputEvent(event)) {\n      return;\n    }\n\n    if (this.expandable) {\n      this.expanded = !this.expanded;\n      this.expandedChange.emit(this.expanded);\n    }\n  }\n\n  getStackChildrenId() {\n    return this.expanded ? `clr-stack-children-${this.uniqueId}` : null;\n  }\n\n  protected preventDefaultIfNotInputEvent(event: Event) {\n    if (eventIsInputEvent(event)) {\n      return;\n    }\n\n    event.preventDefault();\n  }\n}\n\nfunction eventIsInputEvent(event?: Event) {\n  const targetElement = event?.target as HTMLElement;\n\n  return targetElement?.tagName\n    ? ['INPUT', 'TEXTAREA', 'BUTTON', 'A', 'SELECT', 'OPTION'].includes(targetElement.tagName)\n    : false;\n}\n"]}