UNPKG

carbon-components-angular

Version:
480 lines (474 loc) 41.8 kB
import { Component, Input, HostBinding, ViewChild } from "@angular/core"; import { BaseIconButton } from "carbon-components-angular/button"; import * as i0 from "@angular/core"; import * as i1 from "carbon-components-angular/i18n"; import * as i2 from "carbon-components-angular/utils"; import * as i3 from "@angular/common"; import * as i4 from "carbon-components-angular/button"; import * as i5 from "carbon-components-angular/icon"; export var SnippetType; (function (SnippetType) { SnippetType["single"] = "single"; SnippetType["multi"] = "multi"; SnippetType["inline"] = "inline"; })(SnippetType || (SnippetType = {})); /** * Get started with importing the module: * * ```typescript * import { CodeSnippetModule } from 'carbon-components-angular'; * ``` * * ```html * <cds-code-snippet>Code</cds-code-snippet> * ``` * * [See demo](../../?path=/story/components-code-snippet--basic) */ export class CodeSnippet extends BaseIconButton { /** * Creates an instance of CodeSnippet. */ constructor(i18n, eventService) { super(); this.i18n = i18n; this.eventService = eventService; this.rowHeightInPixel = 16; /** * It can be `"single"`, `"multi"` or `"inline"` */ this.display = SnippetType.single; this.translations = this.i18n.get().CODE_SNIPPET; /** * Set to `true` to hide copy button */ this.hideCopyButton = false; /** * Set to `true` to disable the code snippet */ this.disabled = false; /** * Specify the max number of rows to show when collapsed * Default is `15` */ this.maxCollapsedNumberOfRows = 15; /** * Specify the min number of rows to show when collapsed * Default is `3` */ this.minCollapsedNumberOfRows = 3; /** * Specify the max number of rows to show when expanded * Default is `0`, hence all content will be visible when expanded */ this.maxExpandedNumberOfRows = 0; /** * Specify the min number of rows to show when expanded * Default is `16`, hence height of expanded row will be 16 * rowHeightInPixel (16) = 256px */ this.minExpandedNumberOfRows = 16; /** * Set to `true` to wrap the text */ this.wrapText = false; /** * @deprecated since v5 - Use `cdsLayer` directive instead * Set to `"light"` to apply the light style */ this.theme = "dark"; /** * Text displayed in the tooltip when user clicks button to copy code. */ this.feedbackText = this.translations.COPIED; /** * Time in miliseconds to keep the feedback tooltip displayed. * Defaults to 2 seconds */ this.feedbackTimeout = 2000; this.expanded = false; this.skeleton = false; this.styles = {}; this.showFeedback = false; this.animating = false; this.hasExpandButton = null; this.isExpandable = false; this.hasRightOverflow = false; this.hasRight = false; this.hasLeft = false; this.dropShadow = false; } get snippetClass() { return this.display !== SnippetType.inline; } get snippetSingleClass() { return this.display === SnippetType.single; } get snippetMultiClass() { return this.display === SnippetType.multi; } get snippetDisabledClass() { return this.display !== "inline" && this.disabled; } get snippetInlineLightClass() { return this.theme === "light"; } handleScroll() { if (this.skeleton) { return; } let ref; switch (this.display) { case "multi": ref = this.codeContent.nativeElement; break; case "single": ref = this.codeContainer.nativeElement; break; default: return; } if (ref) { const { scrollWidth, clientWidth, scrollLeft } = ref; const horizontalOverflow = scrollWidth > clientWidth; this.hasLeft = horizontalOverflow && !!scrollLeft; this.hasRight = horizontalOverflow && scrollLeft + clientWidth !== scrollWidth; } } toggleSnippetExpansion() { this.expanded = !this.expanded; this.calculateContainerHeight(); } onCopyButtonClicked() { if (!this.disabled) { window.navigator.clipboard .writeText(this.code.nativeElement.innerText || this.code.nativeElement.textContent).then(() => { this.showFeedback = true; this.animating = true; setTimeout(() => { this.showFeedback = false; this.animating = false; }, this.feedbackTimeout); }); } } ngOnInit() { this.calculateContainerHeight(); } ngAfterViewInit() { this.canExpand(); this.handleScroll(); if (window) { this.eventService.on(window, "resize", () => { this.canExpand(); this.handleScroll(); }); } } calculateContainerHeight() { if (this.display === "multi" && !this.skeleton) { this.styles = {}; if (this.expanded) { if (this.maxExpandedNumberOfRows > 0) { this.styles["max-height"] = `${this.maxExpandedNumberOfRows * this.rowHeightInPixel}px`; } if (this.minExpandedNumberOfRows > 0) { this.styles["min-height"] = `${this.minExpandedNumberOfRows * this.rowHeightInPixel}px`; } } else { if (this.maxCollapsedNumberOfRows > 0) { this.styles["max-height"] = `${this.maxCollapsedNumberOfRows * this.rowHeightInPixel}px`; } if (this.minCollapsedNumberOfRows > 0) { this.styles["min-height"] = `${this.minCollapsedNumberOfRows * this.rowHeightInPixel}px`; } } } } canExpand() { if (this.display === "multi" && !this.skeleton) { const height = this.codeContent.nativeElement.getBoundingClientRect().height; if (this.maxCollapsedNumberOfRows > 0 && (this.maxExpandedNumberOfRows <= 0 || this.maxExpandedNumberOfRows > this.maxCollapsedNumberOfRows) && height > this.maxCollapsedNumberOfRows * this.rowHeightInPixel) { this.isExpandable = true; } else { this.isExpandable = false; } if (this.expanded && this.minExpandedNumberOfRows > 0 && height <= this.minExpandedNumberOfRows * this.rowHeightInPixel) { this.isExpandable = false; } } } } CodeSnippet.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CodeSnippet, deps: [{ token: i1.I18n }, { token: i2.EventService }], target: i0.ɵɵFactoryTarget.Component }); CodeSnippet.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: CodeSnippet, selector: "cds-code-snippet, ibm-code-snippet", inputs: { display: "display", translations: "translations", copyButtonDescription: "copyButtonDescription", hideCopyButton: "hideCopyButton", disabled: "disabled", maxCollapsedNumberOfRows: "maxCollapsedNumberOfRows", minCollapsedNumberOfRows: "minCollapsedNumberOfRows", maxExpandedNumberOfRows: "maxExpandedNumberOfRows", minExpandedNumberOfRows: "minExpandedNumberOfRows", wrapText: "wrapText", theme: "theme", feedbackText: "feedbackText", feedbackTimeout: "feedbackTimeout", expanded: "expanded", skeleton: "skeleton" }, host: { properties: { "class.cds--snippet": "this.snippetClass", "class.cds--snippet--single": "this.snippetSingleClass", "class.cds--snippet--multi": "this.snippetMultiClass", "class.cds--snippet--disabled": "this.snippetDisabledClass", "class.cds--snippet--light": "this.snippetInlineLightClass", "class.cds--snippet--wraptext": "this.wrapText", "class.cds--snippet--expand": "this.expanded", "class.cds--skeleton": "this.skeleton" } }, viewQueries: [{ propertyName: "code", first: true, predicate: ["code"], descendants: true }, { propertyName: "codeContent", first: true, predicate: ["codeContent"], descendants: true }, { propertyName: "codeContainer", first: true, predicate: ["codeContainer"], descendants: true }], usesInheritance: true, ngImport: i0, template: ` <ng-container *ngIf="display === 'inline'; else notInline"> <ng-container *ngIf="!hideCopyButton; else noBtnInline"> <ng-container *ngTemplateOutlet="buttonTemplate"></ng-container> </ng-container> <ng-template #noBtnInline> <span class="cds--snippet cds--snippet--inline cds--snippet--no-copy" [ngClass]="{ 'cds--snippet--light': theme === 'light' }"> <code #code> <ng-container *ngTemplateOutlet="codeTemplate"></ng-container> </code> </span> </ng-template> </ng-container> <ng-template #notInline> <div #codeContainer class="cds--snippet-container" [attr.aria-label]="translations.CODE_SNIPPET_TEXT" [attr.tabindex]="display === 'single' && !disabled ? '0' : null" [attr.role]="display==='single' ? 'textarea' : null" [ngStyle]="styles" (scroll)="(display === 'single' ? handleScroll() : null)"> <ng-container *ngIf="skeleton"> <span *ngIf="display === 'single'; else multiSkeleton"></span> <ng-template #multiSkeleton> <span></span> <span></span> <span></span> </ng-template> </ng-container> <pre #codeContent *ngIf="!skeleton" (scroll)="(display === 'multi' ? handleScroll() : null)"><code #code><ng-container *ngTemplateOutlet="codeTemplate"></ng-container></code></pre> </div> <div *ngIf="hasLeft" class="cds--snippet__overflow-indicator--left"></div> <div *ngIf="hasRight" class="cds--snippet__overflow-indicator--right"></div> <ng-container *ngIf="!hideCopyButton;"> <ng-container *ngTemplateOutlet="buttonTemplate"></ng-container> </ng-container> <button *ngIf="isExpandable" class="cds--btn cds--btn--ghost cds--btn--sm cds--snippet-btn--expand" (click)="toggleSnippetExpansion()" type="button"> <span class="cds--snippet-btn--text">{{expanded ? translations.SHOW_LESS : translations.SHOW_MORE}}</span> <svg cdsIcon="chevron--down" size="16" class="cds--icon-chevron--down" [attr.aria-label]="translations.SHOW_MORE_ICON"></svg> </button> </ng-template> <ng-template #buttonTemplate> <cds-icon-button *ngIf="!skeleton" [description]="showFeedback ? feedbackText : copyButtonDescription" [align]="align" [autoAlign]="autoAlign" [dropShadow]="dropShadow" [caret]="caret" [highContrast]="highContrast" [isOpen]="isOpen" [enterDelayMs]="enterDelayMs" [leaveDelayMs]="leaveDelayMs" type="button" kind="primary" size="md" (click)="onCopyButtonClicked($event)" [buttonNgClass]="{ 'cds--snippet--light': theme === 'light', 'cds--snippet--inline': display === 'inline', 'cds--btn--icon-only': display !== 'inline', 'cds--copy-btn': display !== 'inline', 'cds--copy-btn--animating': animating, 'cds--copy-btn--fade-in': showFeedback, 'cds--copy-btn--fade-out': !showFeedback && animating, 'cds--snippet cds--copy': true }" [buttonAttributes]="{ 'aria-label': translations.COPY_CODE, 'aria-live': 'polite', 'tabindex': '0' }"> <ng-container *ngIf="display === 'inline'"> <code #code> <ng-container *ngTemplateOutlet="codeTemplate"></ng-container> </code> </ng-container> <ng-container *ngIf="display !== 'inline'"> <svg cdsIcon="copy" size="16" class="cds--snippet__icon"></svg> </ng-container> </cds-icon-button> </ng-template> <ng-template #codeTemplate> <ng-content></ng-content> </ng-template> `, isInline: true, dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i4.IconButton, selector: "cds-icon-button, ibm-icon-button", inputs: ["buttonNgClass", "buttonAttributes", "buttonId", "kind", "size", "type", "isExpressive", "disabled", "description", "showTooltipWhenDisabled"], outputs: ["click", "focus", "blur", "tooltipClick"] }, { kind: "directive", type: i5.IconDirective, selector: "[cdsIcon], [ibmIcon]", inputs: ["ibmIcon", "cdsIcon", "size", "title", "ariaLabel", "ariaLabelledBy", "ariaHidden", "isFocusable"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CodeSnippet, decorators: [{ type: Component, args: [{ selector: "cds-code-snippet, ibm-code-snippet", template: ` <ng-container *ngIf="display === 'inline'; else notInline"> <ng-container *ngIf="!hideCopyButton; else noBtnInline"> <ng-container *ngTemplateOutlet="buttonTemplate"></ng-container> </ng-container> <ng-template #noBtnInline> <span class="cds--snippet cds--snippet--inline cds--snippet--no-copy" [ngClass]="{ 'cds--snippet--light': theme === 'light' }"> <code #code> <ng-container *ngTemplateOutlet="codeTemplate"></ng-container> </code> </span> </ng-template> </ng-container> <ng-template #notInline> <div #codeContainer class="cds--snippet-container" [attr.aria-label]="translations.CODE_SNIPPET_TEXT" [attr.tabindex]="display === 'single' && !disabled ? '0' : null" [attr.role]="display==='single' ? 'textarea' : null" [ngStyle]="styles" (scroll)="(display === 'single' ? handleScroll() : null)"> <ng-container *ngIf="skeleton"> <span *ngIf="display === 'single'; else multiSkeleton"></span> <ng-template #multiSkeleton> <span></span> <span></span> <span></span> </ng-template> </ng-container> <pre #codeContent *ngIf="!skeleton" (scroll)="(display === 'multi' ? handleScroll() : null)"><code #code><ng-container *ngTemplateOutlet="codeTemplate"></ng-container></code></pre> </div> <div *ngIf="hasLeft" class="cds--snippet__overflow-indicator--left"></div> <div *ngIf="hasRight" class="cds--snippet__overflow-indicator--right"></div> <ng-container *ngIf="!hideCopyButton;"> <ng-container *ngTemplateOutlet="buttonTemplate"></ng-container> </ng-container> <button *ngIf="isExpandable" class="cds--btn cds--btn--ghost cds--btn--sm cds--snippet-btn--expand" (click)="toggleSnippetExpansion()" type="button"> <span class="cds--snippet-btn--text">{{expanded ? translations.SHOW_LESS : translations.SHOW_MORE}}</span> <svg cdsIcon="chevron--down" size="16" class="cds--icon-chevron--down" [attr.aria-label]="translations.SHOW_MORE_ICON"></svg> </button> </ng-template> <ng-template #buttonTemplate> <cds-icon-button *ngIf="!skeleton" [description]="showFeedback ? feedbackText : copyButtonDescription" [align]="align" [autoAlign]="autoAlign" [dropShadow]="dropShadow" [caret]="caret" [highContrast]="highContrast" [isOpen]="isOpen" [enterDelayMs]="enterDelayMs" [leaveDelayMs]="leaveDelayMs" type="button" kind="primary" size="md" (click)="onCopyButtonClicked($event)" [buttonNgClass]="{ 'cds--snippet--light': theme === 'light', 'cds--snippet--inline': display === 'inline', 'cds--btn--icon-only': display !== 'inline', 'cds--copy-btn': display !== 'inline', 'cds--copy-btn--animating': animating, 'cds--copy-btn--fade-in': showFeedback, 'cds--copy-btn--fade-out': !showFeedback && animating, 'cds--snippet cds--copy': true }" [buttonAttributes]="{ 'aria-label': translations.COPY_CODE, 'aria-live': 'polite', 'tabindex': '0' }"> <ng-container *ngIf="display === 'inline'"> <code #code> <ng-container *ngTemplateOutlet="codeTemplate"></ng-container> </code> </ng-container> <ng-container *ngIf="display !== 'inline'"> <svg cdsIcon="copy" size="16" class="cds--snippet__icon"></svg> </ng-container> </cds-icon-button> </ng-template> <ng-template #codeTemplate> <ng-content></ng-content> </ng-template> ` }] }], ctorParameters: function () { return [{ type: i1.I18n }, { type: i2.EventService }]; }, propDecorators: { snippetClass: [{ type: HostBinding, args: ["class.cds--snippet"] }], snippetSingleClass: [{ type: HostBinding, args: ["class.cds--snippet--single"] }], snippetMultiClass: [{ type: HostBinding, args: ["class.cds--snippet--multi"] }], snippetDisabledClass: [{ type: HostBinding, args: ["class.cds--snippet--disabled"] }], snippetInlineLightClass: [{ type: HostBinding, args: ["class.cds--snippet--light"] }], display: [{ type: Input }], translations: [{ type: Input }], copyButtonDescription: [{ type: Input }], hideCopyButton: [{ type: Input }], disabled: [{ type: Input }], maxCollapsedNumberOfRows: [{ type: Input }], minCollapsedNumberOfRows: [{ type: Input }], maxExpandedNumberOfRows: [{ type: Input }], minExpandedNumberOfRows: [{ type: Input }], wrapText: [{ type: HostBinding, args: ["class.cds--snippet--wraptext"] }, { type: Input }], theme: [{ type: Input }], feedbackText: [{ type: Input }], feedbackTimeout: [{ type: Input }], expanded: [{ type: HostBinding, args: ["class.cds--snippet--expand"] }, { type: Input }], skeleton: [{ type: HostBinding, args: ["class.cds--skeleton"] }, { type: Input }], code: [{ type: ViewChild, args: ["code"] }], codeContent: [{ type: ViewChild, args: ["codeContent"] }], codeContainer: [{ type: ViewChild, args: ["codeContainer"] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code-snippet.component.js","sourceRoot":"","sources":["../../../src/code-snippet/code-snippet.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tComponent,\n\tInput,\n\tHostBinding,\n\tViewChild,\n\tAfterViewInit,\n\tOnInit\n} from \"@angular/core\";\n\nimport { I18n } from \"carbon-components-angular/i18n\";\nimport { EventService } from \"carbon-components-angular/utils\";\nimport { BaseIconButton } from \"carbon-components-angular/button\";\n\nexport enum SnippetType {\n\tsingle = \"single\",\n\tmulti = \"multi\",\n\tinline = \"inline\"\n}\n\n/**\n * Get started with importing the module:\n *\n * ```typescript\n * import { CodeSnippetModule } from 'carbon-components-angular';\n * ```\n *\n * ```html\n * <cds-code-snippet>Code</cds-code-snippet>\n * ```\n *\n * [See demo](../../?path=/story/components-code-snippet--basic)\n */\n@Component({\n\tselector: \"cds-code-snippet, ibm-code-snippet\",\n\ttemplate: `\n\t\t<ng-container *ngIf=\"display === 'inline'; else notInline\">\n\t\t\t<ng-container *ngIf=\"!hideCopyButton; else noBtnInline\">\n\t\t\t\t<ng-container *ngTemplateOutlet=\"buttonTemplate\"></ng-container>\n\t\t\t</ng-container>\n\t\t\t<ng-template #noBtnInline>\n\t\t\t\t<span\n\t\t\t\t\tclass=\"cds--snippet cds--snippet--inline cds--snippet--no-copy\"\n\t\t\t\t\t[ngClass]=\"{\n\t\t\t\t\t\t'cds--snippet--light': theme === 'light'\n\t\t\t\t\t}\">\n\t\t\t\t\t<code #code>\n\t\t\t\t\t\t<ng-container *ngTemplateOutlet=\"codeTemplate\"></ng-container>\n\t\t\t\t\t</code>\n\t\t\t\t</span>\n\t\t\t</ng-template>\n\t\t</ng-container>\n\n\t\t<ng-template #notInline>\n\t\t\t<div\n\t\t\t\t#codeContainer\n\t\t\t\tclass=\"cds--snippet-container\"\n\t\t\t\t[attr.aria-label]=\"translations.CODE_SNIPPET_TEXT\"\n\t\t\t\t[attr.tabindex]=\"display === 'single' && !disabled ? '0' : null\"\n\t\t\t\t[attr.role]=\"display==='single' ? 'textarea' : null\"\n\t\t\t\t[ngStyle]=\"styles\"\n\t\t\t\t(scroll)=\"(display === 'single' ? handleScroll() : null)\">\n\t\t\t\t<ng-container *ngIf=\"skeleton\">\n\t\t\t\t\t<span *ngIf=\"display === 'single'; else multiSkeleton\"></span>\n\t\t\t\t\t<ng-template #multiSkeleton>\n\t\t\t\t\t\t<span></span>\n\t\t\t\t\t\t<span></span>\n\t\t\t\t\t\t<span></span>\n\t\t\t\t\t</ng-template>\n\t\t\t\t</ng-container>\n\t\t\t\t<pre\n\t\t\t\t\t#codeContent\n\t\t\t\t\t*ngIf=\"!skeleton\"\n\t\t\t\t\t(scroll)=\"(display === 'multi' ? handleScroll() : null)\"><code #code><ng-container *ngTemplateOutlet=\"codeTemplate\"></ng-container></code></pre>\n\t\t\t</div>\n\t\t\t<div *ngIf=\"hasLeft\" class=\"cds--snippet__overflow-indicator--left\"></div>\n\t\t\t<div *ngIf=\"hasRight\" class=\"cds--snippet__overflow-indicator--right\"></div>\n\t\t\t<ng-container *ngIf=\"!hideCopyButton;\">\n\t\t\t\t<ng-container *ngTemplateOutlet=\"buttonTemplate\"></ng-container>\n\t\t\t</ng-container>\n\t\t\t<button\n\t\t\t\t*ngIf=\"isExpandable\"\n\t\t\t\tclass=\"cds--btn cds--btn--ghost cds--btn--sm cds--snippet-btn--expand\"\n\t\t\t\t(click)=\"toggleSnippetExpansion()\"\n\t\t\t\ttype=\"button\">\n\t\t\t\t<span class=\"cds--snippet-btn--text\">{{expanded ? translations.SHOW_LESS : translations.SHOW_MORE}}</span>\n\t\t\t\t<svg cdsIcon=\"chevron--down\" size=\"16\" class=\"cds--icon-chevron--down\" [attr.aria-label]=\"translations.SHOW_MORE_ICON\"></svg>\n\t\t\t</button>\n\t\t</ng-template>\n\n\t\t<ng-template #buttonTemplate>\n\t\t\t<cds-icon-button\n\t\t\t\t*ngIf=\"!skeleton\"\n\t\t\t\t[description]=\"showFeedback ? feedbackText : copyButtonDescription\"\n\t\t\t\t[align]=\"align\"\n\t\t\t\t[autoAlign]=\"autoAlign\"\n\t\t\t\t[dropShadow]=\"dropShadow\"\n\t\t\t\t[caret]=\"caret\"\n\t\t\t\t[highContrast]=\"highContrast\"\n\t\t\t\t[isOpen]=\"isOpen\"\n\t\t\t\t[enterDelayMs]=\"enterDelayMs\"\n\t\t\t\t[leaveDelayMs]=\"leaveDelayMs\"\n\t\t\t\ttype=\"button\"\n\t\t\t\tkind=\"primary\"\n\t\t\t\tsize=\"md\"\n\t\t\t\t(click)=\"onCopyButtonClicked($event)\"\n\t\t\t\t[buttonNgClass]=\"{\n\t\t\t\t\t'cds--snippet--light': theme === 'light',\n\t\t\t\t\t'cds--snippet--inline': display === 'inline',\n\t\t\t\t\t'cds--btn--icon-only': display !== 'inline',\n\t\t\t\t\t'cds--copy-btn': display !== 'inline',\n\t\t\t\t\t'cds--copy-btn--animating': animating,\n\t\t\t\t\t'cds--copy-btn--fade-in': showFeedback,\n\t\t\t\t\t'cds--copy-btn--fade-out': !showFeedback && animating,\n\t\t\t\t\t'cds--snippet cds--copy': true\n\t\t\t\t}\"\n\t\t\t\t[buttonAttributes]=\"{\n\t\t\t\t\t'aria-label': translations.COPY_CODE,\n\t\t\t\t\t'aria-live': 'polite',\n\t\t\t\t\t'tabindex': '0'\n\t\t\t\t}\">\n\t\t\t\t<ng-container *ngIf=\"display === 'inline'\">\n\t\t\t\t\t<code #code>\n\t\t\t\t\t\t<ng-container *ngTemplateOutlet=\"codeTemplate\"></ng-container>\n\t\t\t\t\t</code>\n\t\t\t\t</ng-container>\n\t\t\t\t<ng-container *ngIf=\"display !== 'inline'\">\n\t\t\t\t\t<svg cdsIcon=\"copy\" size=\"16\" class=\"cds--snippet__icon\"></svg>\n\t\t\t\t</ng-container>\n\t\t\t</cds-icon-button>\n\t\t</ng-template>\n\n\t\t<ng-template #codeTemplate>\n\t\t\t<ng-content></ng-content>\n\t\t</ng-template>\n\t`\n})\nexport class CodeSnippet extends BaseIconButton implements OnInit, AfterViewInit {\n\t@HostBinding(\"class.cds--snippet\") get snippetClass() {\n\t\treturn this.display !== SnippetType.inline;\n\t}\n\t@HostBinding(\"class.cds--snippet--single\") get snippetSingleClass() {\n\t\treturn this.display === SnippetType.single;\n\t}\n\t@HostBinding(\"class.cds--snippet--multi\") get snippetMultiClass() {\n\t\treturn this.display === SnippetType.multi;\n\t}\n\t@HostBinding(\"class.cds--snippet--disabled\") get snippetDisabledClass() {\n\t\treturn this.display !== \"inline\" && this.disabled;\n\t}\n\t@HostBinding(\"class.cds--snippet--light\") get snippetInlineLightClass() {\n\t\treturn this.theme === \"light\";\n\t}\n\n\treadonly rowHeightInPixel: number = 16;\n\n\t/**\n\t * It can be `\"single\"`, `\"multi\"` or `\"inline\"`\n\t */\n\t@Input() display: SnippetType = SnippetType.single;\n\t@Input() translations = this.i18n.get().CODE_SNIPPET;\n\t/**\n\t * copy button description to show on hover\n\t */\n\t@Input() copyButtonDescription: string;\n\t/**\n\t * Set to `true` to hide copy button\n\t */\n\t@Input() hideCopyButton = false;\n\t/**\n\t * Set to `true` to disable the code snippet\n\t */\n\t@Input() disabled = false;\n\t/**\n\t * Specify the max number of rows to show when collapsed\n\t * Default is `15`\n\t */\n\t@Input() maxCollapsedNumberOfRows = 15;\n\t/**\n\t * Specify the min number of rows to show when collapsed\n\t * Default is `3`\n\t */\n\t@Input() minCollapsedNumberOfRows = 3;\n\t/**\n\t * Specify the max number of rows to show when expanded\n\t * Default is `0`, hence all content will be visible when expanded\n\t */\n\t@Input() maxExpandedNumberOfRows = 0;\n\t/**\n\t * Specify the min number of rows to show when expanded\n\t * Default is `16`, hence height of expanded row will be 16 * rowHeightInPixel (16) =  256px\n\t */\n\t@Input() minExpandedNumberOfRows = 16;\n\t/**\n\t * Set to `true` to wrap the text\n\t */\n\t@HostBinding(\"class.cds--snippet--wraptext\") @Input() wrapText = false;\n\t/**\n\t * @deprecated since v5 - Use `cdsLayer` directive instead\n\t * Set to `\"light\"` to apply the light style\n\t */\n\t@Input() theme: \"light\" | \"dark\" = \"dark\";\n\t/**\n\t * Text displayed in the tooltip when user clicks button to copy code.\n\t */\n\t@Input() feedbackText = this.translations.COPIED;\n\t/**\n\t * Time in miliseconds to keep the feedback tooltip displayed.\n\t * Defaults to 2 seconds\n\t */\n\t@Input() feedbackTimeout = 2000;\n\n\t@HostBinding(\"class.cds--snippet--expand\") @Input() expanded = false;\n\t@HostBinding(\"class.cds--skeleton\") @Input() skeleton = false;\n\tstyles: any = {};\n\n\t@ViewChild(\"code\") code;\n\t@ViewChild(\"codeContent\") codeContent;\n\t@ViewChild(\"codeContainer\") codeContainer;\n\n\tshowFeedback = false;\n\tanimating = false;\n\thasExpandButton = null;\n\tisExpandable = false;\n\thasRightOverflow = false;\n\n\thasRight = false;\n\thasLeft = false;\n\n\t/**\n\t * Creates an instance of CodeSnippet.\n\t */\n\tconstructor(protected i18n: I18n, protected eventService: EventService) {\n\t\tsuper();\n\t\tthis.dropShadow = false;\n\t}\n\n\thandleScroll() {\n\t\tif (this.skeleton) {\n\t\t\treturn;\n\t\t}\n\n\t\tlet ref;\n\t\tswitch (this.display) {\n\t\t\tcase \"multi\":\n\t\t\t\tref = this.codeContent.nativeElement;\n\t\t\t\tbreak;\n\t\t\tcase \"single\":\n\t\t\t\tref = this.codeContainer.nativeElement;\n\t\t\t\tbreak;\n\t\t\tdefault:\n\t\t\t\treturn;\n\t\t}\n\t\tif (ref) {\n\t\t\tconst {\n\t\t\t\tscrollWidth,\n\t\t\t\tclientWidth,\n\t\t\t\tscrollLeft\n\t\t\t} = ref;\n\t\t\tconst horizontalOverflow = scrollWidth > clientWidth;\n\t\t\tthis.hasLeft = horizontalOverflow && !!scrollLeft;\n\t\t\tthis.hasRight = horizontalOverflow && scrollLeft + clientWidth !== scrollWidth;\n\t\t}\n\t}\n\n\ttoggleSnippetExpansion() {\n\t\tthis.expanded = !this.expanded;\n\t\tthis.calculateContainerHeight();\n\t}\n\n\tonCopyButtonClicked() {\n\t\tif (!this.disabled) {\n\t\t\twindow.navigator.clipboard\n\t\t\t\t.writeText(this.code.nativeElement.innerText || this.code.nativeElement.textContent).then(() => {\n\t\t\t\t\tthis.showFeedback = true;\n\t\t\t\t\tthis.animating = true;\n\t\t\t\t\tsetTimeout(() => {\n\t\t\t\t\t\tthis.showFeedback = false;\n\t\t\t\t\t\tthis.animating = false;\n\t\t\t\t\t}, this.feedbackTimeout);\n\t\t\t\t});\n\t\t}\n\t}\n\n\tngOnInit() {\n\t\tthis.calculateContainerHeight();\n\t}\n\n\tngAfterViewInit() {\n\t\tthis.canExpand();\n\t\tthis.handleScroll();\n\t\tif (window) {\n\t\t\tthis.eventService.on(window as any, \"resize\", () => {\n\t\t\t\tthis.canExpand();\n\t\t\t\tthis.handleScroll();\n\t\t\t});\n\t\t}\n\t}\n\n\tcalculateContainerHeight() {\n\t\tif (this.display === \"multi\" && !this.skeleton) {\n\t\t\tthis.styles = {};\n\t\t\tif (this.expanded) {\n\t\t\t\tif (this.maxExpandedNumberOfRows > 0) {\n\t\t\t\t\tthis.styles[\"max-height\"] = `${this.maxExpandedNumberOfRows * this.rowHeightInPixel}px`;\n\t\t\t\t}\n\t\t\t\tif (this.minExpandedNumberOfRows > 0) {\n\t\t\t\t\tthis.styles[\"min-height\"] = `${this.minExpandedNumberOfRows * this.rowHeightInPixel}px`;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tif (this.maxCollapsedNumberOfRows > 0) {\n\t\t\t\t\tthis.styles[\"max-height\"] = `${this.maxCollapsedNumberOfRows * this.rowHeightInPixel}px`;\n\t\t\t\t}\n\t\t\t\tif (this.minCollapsedNumberOfRows > 0) {\n\t\t\t\t\tthis.styles[\"min-height\"] = `${this.minCollapsedNumberOfRows * this.rowHeightInPixel}px`;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected canExpand() {\n\t\tif (this.display === \"multi\" && !this.skeleton) {\n\t\t\tconst height = this.codeContent.nativeElement.getBoundingClientRect().height;\n\t\t\tif (\n\t\t\t\tthis.maxCollapsedNumberOfRows > 0 &&\n\t\t\t\t(this.maxExpandedNumberOfRows <= 0 ||\n\t\t\t\t\tthis.maxExpandedNumberOfRows > this.maxCollapsedNumberOfRows) &&\n\t\t\t\theight > this.maxCollapsedNumberOfRows * this.rowHeightInPixel\n\t\t\t) {\n\t\t\t\tthis.isExpandable = true;\n\t\t\t} else {\n\t\t\t\tthis.isExpandable = false;\n\t\t\t}\n\n\t\t\tif (\n\t\t\t\tthis.expanded &&\n\t\t\t\tthis.minExpandedNumberOfRows > 0 &&\n\t\t\t\theight <= this.minExpandedNumberOfRows * this.rowHeightInPixel\n\t\t\t) {\n\t\t\t\tthis.isExpandable = false;\n\t\t\t}\n\t\t}\n\t}\n}\n"]}