@dotglitch/ngx-common 1.1.19 → 1.1.20

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,57 @@
1
+ import { ElementRef, EventEmitter, TemplateRef } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ type CSSUnitString = 'px' | '%' | 'em' | 'in' | '';
4
+ type CSSUnit = `${number}${CSSUnitString}` | `var(--${string})`;
5
+ type CSSString = CSSUnit | `calc(${CSSUnit | ''}${'' | ' '}${'+' | '-' | '/' | '*'}${'' | ' '}${CSSUnit | ''})`;
6
+ export declare class ParallaxCardComponent {
7
+ private readonly element;
8
+ content: TemplateRef<ElementRef>;
9
+ background: TemplateRef<ElementRef>;
10
+ backContent: TemplateRef<ElementRef>;
11
+ backBackground: TemplateRef<ElementRef>;
12
+ /**
13
+ *
14
+ */
15
+ loaded: EventEmitter<any>;
16
+ /**
17
+ * Width of the card
18
+ * @default `240px`
19
+ */
20
+ width: CSSString;
21
+ /**
22
+ * Height of the card
23
+ * @default `320px`
24
+ */
25
+ height: CSSString;
26
+ /**
27
+ * Inset padding of the parallax
28
+ * @default 80
29
+ */
30
+ bgInset: number;
31
+ /**
32
+ * Duration for the flip animation in ms
33
+ * @default 80
34
+ */
35
+ flipAnimationDuration: number;
36
+ renderCardFront: boolean;
37
+ renderCardBack: boolean;
38
+ showBackOfCard: boolean;
39
+ private get wrapper();
40
+ private get cardFront();
41
+ private get cardBack();
42
+ private get backgroundElement();
43
+ private get backfaceBackgroundElement();
44
+ private pointerX;
45
+ private pointerY;
46
+ private pointerLeave;
47
+ constructor(element: ElementRef);
48
+ ngAfterViewInit(): void;
49
+ onPointerMove(e: PointerEvent): void;
50
+ onPointerEnter(): void;
51
+ onPointerLeave(): void;
52
+ onClick(): void;
53
+ render: () => void;
54
+ static ɵfac: i0.ɵɵFactoryDeclaration<ParallaxCardComponent, never>;
55
+ static ɵcmp: i0.ɵɵComponentDeclaration<ParallaxCardComponent, "ngx-parallax-card", never, { "width": { "alias": "width"; "required": false; }; "height": { "alias": "height"; "required": false; }; "bgInset": { "alias": "bgInset"; "required": false; }; "flipAnimationDuration": { "alias": "flipAnimationDuration"; "required": false; }; }, { "loaded": "loaded"; }, ["content", "background", "backContent", "backBackground"], ["*"], true, never>;
56
+ }
57
+ export {};
@@ -0,0 +1,135 @@
1
+ import { NgTemplateOutlet } from '@angular/common';
2
+ import { Component, ContentChild, EventEmitter, Input, Output, TemplateRef } from '@angular/core';
3
+ import * as i0 from "@angular/core";
4
+ export class ParallaxCardComponent {
5
+ get wrapper() { return this.element.nativeElement; }
6
+ get cardFront() { return this.wrapper.querySelector('.card.front'); }
7
+ get cardBack() { return this.wrapper.querySelector('.card.backface'); }
8
+ get backgroundElement() { return this.cardFront.querySelector('.card-bg'); }
9
+ get backfaceBackgroundElement() { return this.cardBack.querySelector('.card-bg'); }
10
+ constructor(element) {
11
+ this.element = element;
12
+ /**
13
+ *
14
+ */
15
+ this.loaded = new EventEmitter();
16
+ /**
17
+ * Width of the card
18
+ * @default `240px`
19
+ */
20
+ this.width = '240px';
21
+ /**
22
+ * Height of the card
23
+ * @default `320px`
24
+ */
25
+ this.height = '320px';
26
+ /**
27
+ * Inset padding of the parallax
28
+ * @default 80
29
+ */
30
+ this.bgInset = 80;
31
+ /**
32
+ * Duration for the flip animation in ms
33
+ * @default 80
34
+ */
35
+ this.flipAnimationDuration = 1600;
36
+ this.renderCardFront = true;
37
+ this.renderCardBack = true;
38
+ this.showBackOfCard = false;
39
+ this.pointerX = 0;
40
+ this.pointerY = 0;
41
+ this.pointerLeave = 0;
42
+ this.render = () => {
43
+ const { width, height } = this.wrapper.getBoundingClientRect();
44
+ const mousePX = this.pointerX / width;
45
+ const mousePY = this.pointerY / height;
46
+ // Rotation factors
47
+ const rX = mousePX * this.bgInset / 1.75;
48
+ const rY = mousePY * -this.bgInset / 1.75;
49
+ // Translation factors
50
+ const tX = mousePX * -this.bgInset * 2;
51
+ const tY = mousePY * -this.bgInset * 2;
52
+ if (this.renderCardFront) {
53
+ this.backgroundElement.style.transform = `translateX(${tX}px) translateY(${tY}px)`;
54
+ }
55
+ if (this.renderCardBack) {
56
+ this.backfaceBackgroundElement.style.transform = `translateX(${tX}px) translateY(${tY}px)`;
57
+ }
58
+ if (this.showBackOfCard) {
59
+ this.cardFront.style.transform = `rotateY(180deg) rotateX(${-rY}deg)`;
60
+ this.cardBack.style.transform = `rotateY(${-rX}deg) rotateX(${-rY}deg)`;
61
+ }
62
+ else {
63
+ this.cardFront.style.transform = `rotateY(${rX}deg) rotateX(${rY}deg)`;
64
+ this.cardBack.style.transform = `rotateY(180deg) rotateX(${-rY}deg)`;
65
+ }
66
+ };
67
+ }
68
+ ngAfterViewInit() {
69
+ const el = this.wrapper;
70
+ // Directly attach events to the wrapper
71
+ el.onpointermove = (e) => this.onPointerMove(e);
72
+ el.onpointerenter = () => this.onPointerEnter();
73
+ el.onpointerleave = () => this.onPointerLeave();
74
+ el.onclick = () => this.onClick();
75
+ this.loaded.emit();
76
+ }
77
+ onPointerMove(e) {
78
+ const { width, height, left, top } = this.wrapper.getBoundingClientRect();
79
+ this.pointerX = e.pageX - left - (width / 2);
80
+ this.pointerY = e.pageY - top - (height / 2);
81
+ this.render();
82
+ }
83
+ onPointerEnter() {
84
+ clearTimeout(this.pointerLeave);
85
+ }
86
+ onPointerLeave() {
87
+ this.pointerLeave = setTimeout(() => {
88
+ this.pointerX = 0;
89
+ this.pointerY = 0;
90
+ this.render();
91
+ }, 600);
92
+ }
93
+ // TODO: This can get intercepted in some states
94
+ onClick() {
95
+ this.showBackOfCard = !this.showBackOfCard;
96
+ this.render();
97
+ }
98
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ParallaxCardComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
99
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.1.2", type: ParallaxCardComponent, isStandalone: true, selector: "ngx-parallax-card", inputs: { width: "width", height: "height", bgInset: "bgInset", flipAnimationDuration: "flipAnimationDuration" }, outputs: { loaded: "loaded" }, host: { properties: { "style.width": "width", "style.height": "height", "style.--card-bg-inset": "-bgInset+\"px\"", "style.--flip-animation-duration": "flipAnimationDuration+\"ms\"", "class.flip": "showBackOfCard" } }, queries: [{ propertyName: "content", first: true, predicate: ["content"], descendants: true, read: TemplateRef }, { propertyName: "background", first: true, predicate: ["background"], descendants: true, read: TemplateRef }, { propertyName: "backContent", first: true, predicate: ["backContent"], descendants: true, read: TemplateRef }, { propertyName: "backBackground", first: true, predicate: ["backBackground"], descendants: true, read: TemplateRef }], ngImport: i0, template: "@if (renderCardBack) {\n <div class=\"card backface\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"backBackground\" />\n </div>\n <div class=\"card-content\">\n <ng-template [ngTemplateOutlet]=\"backContent\" />\n </div>\n </div>\n}\n\n@if (renderCardFront) {\n <div class=\"card front\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"background\" />\n </div>\n <div class=\"card-content\">\n @if (content) {\n <ng-template [ngTemplateOutlet]=\"content\"/>\n }\n @else {\n <ng-content/>\n }\n </div>\n </div>\n}\n\n\n", styles: [":host{display:block;position:relative;cursor:pointer;--easing-function: cubic-bezier(.23, 1, .32, 1);--border-radius: 9px}.card{position:absolute;height:100%;width:100%;background-color:#333;backface-visibility:hidden;overflow:hidden;box-shadow:#0000 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px;border-radius:var(--border-radius);transition:transform var(--flip-animation-duration) var(--easing-function),box-shadow 2s var(--easing-function);transform-style:preserve-3d}.card:hover{box-shadow:#fff6 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px}.card:hover .card-content{border:2px solid #fff}.card-bg{inset:var(--card-bg-inset);position:absolute;pointer-events:none;transition:transform .6s var(--easing-function)}.card-content{position:absolute;top:0;color:#fff;height:100%;width:100%;z-index:1;border:2px solid rgba(255,255,255,0);border-radius:var(--border-radius);transition:transform .6s var(--easing-function),border-color .6s var(--easing-function)}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }] }); }
100
+ }
101
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ParallaxCardComponent, decorators: [{
102
+ type: Component,
103
+ args: [{ selector: 'ngx-parallax-card', imports: [
104
+ NgTemplateOutlet
105
+ ], host: {
106
+ '[style.width]': 'width',
107
+ '[style.height]': 'height',
108
+ '[style.--card-bg-inset]': '-bgInset+"px"',
109
+ '[style.--flip-animation-duration]': 'flipAnimationDuration+"ms"',
110
+ '[class.flip]': 'showBackOfCard'
111
+ }, standalone: true, template: "@if (renderCardBack) {\n <div class=\"card backface\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"backBackground\" />\n </div>\n <div class=\"card-content\">\n <ng-template [ngTemplateOutlet]=\"backContent\" />\n </div>\n </div>\n}\n\n@if (renderCardFront) {\n <div class=\"card front\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"background\" />\n </div>\n <div class=\"card-content\">\n @if (content) {\n <ng-template [ngTemplateOutlet]=\"content\"/>\n }\n @else {\n <ng-content/>\n }\n </div>\n </div>\n}\n\n\n", styles: [":host{display:block;position:relative;cursor:pointer;--easing-function: cubic-bezier(.23, 1, .32, 1);--border-radius: 9px}.card{position:absolute;height:100%;width:100%;background-color:#333;backface-visibility:hidden;overflow:hidden;box-shadow:#0000 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px;border-radius:var(--border-radius);transition:transform var(--flip-animation-duration) var(--easing-function),box-shadow 2s var(--easing-function);transform-style:preserve-3d}.card:hover{box-shadow:#fff6 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px}.card:hover .card-content{border:2px solid #fff}.card-bg{inset:var(--card-bg-inset);position:absolute;pointer-events:none;transition:transform .6s var(--easing-function)}.card-content{position:absolute;top:0;color:#fff;height:100%;width:100%;z-index:1;border:2px solid rgba(255,255,255,0);border-radius:var(--border-radius);transition:transform .6s var(--easing-function),border-color .6s var(--easing-function)}\n"] }]
112
+ }], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { content: [{
113
+ type: ContentChild,
114
+ args: ['content', { read: TemplateRef }]
115
+ }], background: [{
116
+ type: ContentChild,
117
+ args: ['background', { read: TemplateRef }]
118
+ }], backContent: [{
119
+ type: ContentChild,
120
+ args: ['backContent', { read: TemplateRef }]
121
+ }], backBackground: [{
122
+ type: ContentChild,
123
+ args: ['backBackground', { read: TemplateRef }]
124
+ }], loaded: [{
125
+ type: Output
126
+ }], width: [{
127
+ type: Input
128
+ }], height: [{
129
+ type: Input
130
+ }], bgInset: [{
131
+ type: Input
132
+ }], flipAnimationDuration: [{
133
+ type: Input
134
+ }] } });
135
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parallax-card.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/parallax-card/parallax-card.component.ts","../../../../../packages/common/src/components/parallax-card/parallax-card.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgTemplateOutlet } from '@angular/common';\nimport { Component, ContentChild, ElementRef, EventEmitter, Input, Output, TemplateRef } from '@angular/core';\n\ntype CSSUnitString = 'px' | '%' | 'em' | 'in' | '';\ntype CSSUnit = `${number}${CSSUnitString}` | `var(--${string})`;\ntype CSSString = CSSUnit |\n    `calc(${CSSUnit|''}${''|' '}${'+'|'-'|'/'|'*'}${''|' '}${CSSUnit|''})`\n\n@Component({\n    selector: 'ngx-parallax-card',\n    templateUrl: './parallax-card.component.html',\n    styleUrls: ['./parallax-card.component.scss'],\n    imports: [\n        NgTemplateOutlet\n    ],\n    host: {\n        '[style.width]': 'width',\n        '[style.height]': 'height',\n        '[style.--card-bg-inset]': '-bgInset+\"px\"',\n        '[style.--flip-animation-duration]': 'flipAnimationDuration+\"ms\"',\n        '[class.flip]': 'showBackOfCard'\n    },\n    standalone: true\n})\nexport class ParallaxCardComponent {\n\n    // Front of card\n    @ContentChild('content', { read: TemplateRef }) content: TemplateRef<ElementRef>;\n    @ContentChild('background', { read: TemplateRef }) background: TemplateRef<ElementRef>;\n\n    // Back of card\n    @ContentChild('backContent', { read: TemplateRef }) backContent: TemplateRef<ElementRef>;\n    @ContentChild('backBackground', { read: TemplateRef }) backBackground: TemplateRef<ElementRef>;\n\n    /**\n     *\n     */\n    @Output() loaded = new EventEmitter();\n\n    /**\n     * Width of the card\n     * @default `240px`\n     */\n    @Input() width:  CSSString = '240px';\n    /**\n     * Height of the card\n     * @default `320px`\n     */\n    @Input() height: CSSString = '320px';\n    /**\n     * Inset padding of the parallax\n     * @default 80\n     */\n    @Input() bgInset:  number = 80;\n    /**\n     * Duration for the flip animation in ms\n     * @default 80\n     */\n    @Input() flipAnimationDuration:  number = 1600;\n\n    renderCardFront = true;\n    renderCardBack = true;\n    showBackOfCard = false;\n\n    private get wrapper() { return this.element.nativeElement as HTMLElement; }\n    private get cardFront() { return this.wrapper.querySelector('.card.front') as HTMLElement; }\n    private get cardBack() { return this.wrapper.querySelector('.card.backface') as HTMLElement; }\n    private get backgroundElement() { return this.cardFront.querySelector('.card-bg') as HTMLDivElement; }\n    private get backfaceBackgroundElement() { return this.cardBack.querySelector('.card-bg') as HTMLDivElement; }\n\n    private pointerX = 0;\n    private pointerY = 0;\n    private pointerLeave = 0;\n\n    constructor(\n        private readonly element: ElementRef\n    ) { }\n\n    ngAfterViewInit() {\n        const el = this.wrapper;\n\n        // Directly attach events to the wrapper\n        el.onpointermove = (e) => this.onPointerMove(e);\n        el.onpointerenter = () => this.onPointerEnter();\n        el.onpointerleave = () => this.onPointerLeave();\n        el.onclick = () => this.onClick();\n\n        this.loaded.emit();\n    }\n\n    onPointerMove(e: PointerEvent) {\n        const { width, height, left, top } = this.wrapper.getBoundingClientRect();\n        this.pointerX = e.pageX - left - (width / 2);\n        this.pointerY = e.pageY - top - (height / 2);\n\n        this.render();\n    }\n\n    onPointerEnter() {\n        clearTimeout(this.pointerLeave);\n    }\n\n    onPointerLeave() {\n        this.pointerLeave = setTimeout(() => {\n            this.pointerX = 0;\n            this.pointerY = 0;\n            this.render();\n        }, 600) as any;\n    }\n\n    // TODO: This can get intercepted in some states\n    onClick() {\n        this.showBackOfCard = !this.showBackOfCard;\n        this.render()\n    }\n\n    render = () => {\n        const { width, height } = this.wrapper.getBoundingClientRect();\n        const mousePX = this.pointerX / width;\n        const mousePY = this.pointerY / height;\n\n        // Rotation factors\n        const rX = mousePX * this.bgInset / 1.75;\n        const rY = mousePY * -this.bgInset / 1.75;\n\n        // Translation factors\n        const tX = mousePX * -this.bgInset * 2;\n        const tY = mousePY * -this.bgInset * 2;\n\n        if (this.renderCardFront) {\n            this.backgroundElement.style.transform = `translateX(${tX}px) translateY(${tY}px)`;\n        }\n        if (this.renderCardBack) {\n            this.backfaceBackgroundElement.style.transform = `translateX(${tX}px) translateY(${tY}px)`;\n        }\n\n        if (this.showBackOfCard) {\n            this.cardFront.style.transform = `rotateY(180deg) rotateX(${-rY}deg)`;\n            this.cardBack.style.transform = `rotateY(${-rX}deg) rotateX(${-rY}deg)`;\n        }\n        else {\n            this.cardFront.style.transform = `rotateY(${rX}deg) rotateX(${rY}deg)`;\n            this.cardBack.style.transform = `rotateY(180deg) rotateX(${-rY}deg)`;\n        }\n    }\n}\n","@if (renderCardBack) {\n    <div class=\"card backface\">\n        <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n            <ng-template [ngTemplateOutlet]=\"backBackground\" />\n        </div>\n        <div class=\"card-content\">\n            <ng-template [ngTemplateOutlet]=\"backContent\" />\n        </div>\n    </div>\n}\n\n@if (renderCardFront) {\n    <div class=\"card front\">\n        <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n            <ng-template [ngTemplateOutlet]=\"background\" />\n        </div>\n        <div class=\"card-content\">\n            @if (content) {\n                <ng-template [ngTemplateOutlet]=\"content\"/>\n            }\n            @else {\n                <ng-content/>\n            }\n        </div>\n    </div>\n}\n\n\n"]}
@@ -53,7 +53,8 @@ export * from './components/filemanager/filemanager.component';
53
53
  // export * from './components/music-library/music-library.component';
54
54
  export * from './components/tabulator/tabulator.component';
55
55
  export * from './components/vscode/vscode.component';
56
+ export * from './components/parallax-card/parallax-card.component';
56
57
  export * from './components/react-magic-wrapper/react-magic-wrapper.component';
57
58
  export * from './components/types';
58
59
  export { ConsoleLogger, LogIcon } from './utils/index';
59
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL2NvbW1vbi9zcmMvcHVibGljLWFwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVIOztHQUVHO0FBQ0gsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxlQUFlLENBQUM7QUFFOUI7O0dBRUc7QUFDSCxjQUFjLGdDQUFnQyxDQUFDO0FBQy9DLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYyxvQ0FBb0MsQ0FBQztBQUVuRDs7R0FFRztBQUNILGNBQWMsMEJBQTBCLENBQUM7QUFDekMsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLDRCQUE0QixDQUFDO0FBQzNDLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyx5QkFBeUIsQ0FBQztBQUV4Qzs7RUFFRTtBQUNGLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyxvQ0FBb0MsQ0FBQztBQUNuRCxvREFBb0Q7QUFFcEQ7O0VBRUU7QUFDRixjQUFjLDhDQUE4QyxDQUFDO0FBQzdELGNBQWMsNkNBQTZDLENBQUM7QUFDNUQsY0FBYyxnREFBZ0QsQ0FBQztBQUMvRCxjQUFjLGdDQUFnQyxDQUFDO0FBRS9DOztFQUVFO0FBQ0YsY0FBYyxnREFBZ0QsQ0FBQztBQUMvRCxjQUFjLCtDQUErQyxDQUFDO0FBQzlELGNBQWMsa0RBQWtELENBQUM7QUFDakUsY0FBYyxpQ0FBaUMsQ0FBQztBQUVoRDs7R0FFRztBQUNILGNBQWMsZ0RBQWdELENBQUM7QUFDL0Qsc0VBQXNFO0FBQ3RFLGNBQWMsNENBQTRDLENBQUM7QUFDM0QsY0FBYyxzQ0FBc0MsQ0FBQztBQUNyRCxjQUFjLGdFQUFnRSxDQUFDO0FBQy9FLGNBQWMsb0JBQW9CLENBQUM7QUFHbkMsT0FBTyxFQUFFLGFBQWEsRUFBRSxPQUFPLEVBQUUsTUFBTSxlQUFlLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIHBhY2thZ2VcbiAqL1xuXG4vKipcbiAqKiBUeXBlc1xuICovXG5leHBvcnQgKiBmcm9tICcuL3R5cGVzL21lbnUnO1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcy9wb3B1cCc7XG5cbi8qKlxuICoqIERpcmVjdGl2ZXNcbiAqL1xuZXhwb3J0ICogZnJvbSAnLi9kaXJlY3RpdmVzL3Rvb2x0aXAuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vZGlyZWN0aXZlcy9tZW51LmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL2RpcmVjdGl2ZXMvaW1hZ2UtY2FjaGUuZGlyZWN0aXZlJztcblxuLyoqXG4gKiogUGlwZXNcbiAqL1xuZXhwb3J0ICogZnJvbSAnLi9waXBlcy9odG1sLWJ5cGFzcy5waXBlJztcbmV4cG9ydCAqIGZyb20gJy4vcGlwZXMvcmVzb3VyY2UtYnlwYXNzLnBpcGUnO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlcy9zY3JpcHQtYnlwYXNzLnBpcGUnO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlcy9zdHlsZS1ieXBhc3MucGlwZSc7XG5leHBvcnQgKiBmcm9tICcuL3BpcGVzL3VybC1ieXBhc3MucGlwZSc7XG5cbi8qKlxuKiogU2VydmljZXNcbiovXG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzL2RlcGVuZGVuY3kuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzL2RpYWxvZy5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZXMvZmV0Y2guc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzL2tleWJvYXJkLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcy9maWxlLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcy90aGVtZS5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZXMvbmF2aWdhdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZXMvY29tbWFuZC1wYWxldHRlLnNlcnZpY2UnO1xuLy8gZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcy9zZXJ2aWNld29ya2VyLnNlcnZpY2UnO1xuXG4vKipcbioqIExhenkgbG9hZGVyIGNvbXBvbmVudCAmIHNlcnZpY2VcbiovXG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvbGF6eS1sb2FkZXIvbGF6eS1sb2FkZXIuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvbGF6eS1sb2FkZXIvbGF6eS1sb2FkZXIubW9kdWxlJztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9sYXp5LWxvYWRlci9sYXp5LWxvYWRlci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL2xhenktbG9hZGVyL3R5cGVzJztcblxuLyoqXG4qKiBEeW5hbWljIEhUTUwgKFdJUClcbiovXG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvZHluYW1pYy1odG1sL2R5bmFtaWMtaHRtbC5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9keW5hbWljLWh0bWwvZHluYW1pYy1odG1sLm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvZHluYW1pYy1odG1sL2R5bmFtaWMtaHRtbC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL2R5bmFtaWMtaHRtbC90eXBlcyc7XG5cbi8qKlxuICoqIENvbXBvbmVudHNcbiAqL1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL2ZpbGVtYW5hZ2VyL2ZpbGVtYW5hZ2VyLmNvbXBvbmVudCc7XG4vLyBleHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvbXVzaWMtbGlicmFyeS9tdXNpYy1saWJyYXJ5LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvdGFidWxhdG9yL3RhYnVsYXRvci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL3ZzY29kZS92c2NvZGUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9yZWFjdC1tYWdpYy13cmFwcGVyL3JlYWN0LW1hZ2ljLXdyYXBwZXIuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy90eXBlcyc7XG5cblxuZXhwb3J0IHsgQ29uc29sZUxvZ2dlciwgTG9nSWNvbiB9IGZyb20gJy4vdXRpbHMvaW5kZXgnO1xuIl19
60
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL2NvbW1vbi9zcmMvcHVibGljLWFwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVIOztHQUVHO0FBQ0gsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxlQUFlLENBQUM7QUFFOUI7O0dBRUc7QUFDSCxjQUFjLGdDQUFnQyxDQUFDO0FBQy9DLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYyxvQ0FBb0MsQ0FBQztBQUVuRDs7R0FFRztBQUNILGNBQWMsMEJBQTBCLENBQUM7QUFDekMsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLDRCQUE0QixDQUFDO0FBQzNDLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyx5QkFBeUIsQ0FBQztBQUV4Qzs7RUFFRTtBQUNGLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyxvQ0FBb0MsQ0FBQztBQUNuRCxvREFBb0Q7QUFFcEQ7O0VBRUU7QUFDRixjQUFjLDhDQUE4QyxDQUFDO0FBQzdELGNBQWMsNkNBQTZDLENBQUM7QUFDNUQsY0FBYyxnREFBZ0QsQ0FBQztBQUMvRCxjQUFjLGdDQUFnQyxDQUFDO0FBRS9DOztFQUVFO0FBQ0YsY0FBYyxnREFBZ0QsQ0FBQztBQUMvRCxjQUFjLCtDQUErQyxDQUFDO0FBQzlELGNBQWMsa0RBQWtELENBQUM7QUFDakUsY0FBYyxpQ0FBaUMsQ0FBQztBQUVoRDs7R0FFRztBQUNILGNBQWMsZ0RBQWdELENBQUM7QUFDL0Qsc0VBQXNFO0FBQ3RFLGNBQWMsNENBQTRDLENBQUM7QUFDM0QsY0FBYyxzQ0FBc0MsQ0FBQztBQUNyRCxjQUFjLG9EQUFvRCxDQUFDO0FBQ25FLGNBQWMsZ0VBQWdFLENBQUM7QUFDL0UsY0FBYyxvQkFBb0IsQ0FBQztBQUduQyxPQUFPLEVBQUUsYUFBYSxFQUFFLE9BQU8sRUFBRSxNQUFNLGVBQWUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgcGFja2FnZVxuICovXG5cbi8qKlxuICoqIFR5cGVzXG4gKi9cbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMvbWVudSc7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzL3BvcHVwJztcblxuLyoqXG4gKiogRGlyZWN0aXZlc1xuICovXG5leHBvcnQgKiBmcm9tICcuL2RpcmVjdGl2ZXMvdG9vbHRpcC5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi9kaXJlY3RpdmVzL21lbnUuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vZGlyZWN0aXZlcy9pbWFnZS1jYWNoZS5kaXJlY3RpdmUnO1xuXG4vKipcbiAqKiBQaXBlc1xuICovXG5leHBvcnQgKiBmcm9tICcuL3BpcGVzL2h0bWwtYnlwYXNzLnBpcGUnO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlcy9yZXNvdXJjZS1ieXBhc3MucGlwZSc7XG5leHBvcnQgKiBmcm9tICcuL3BpcGVzL3NjcmlwdC1ieXBhc3MucGlwZSc7XG5leHBvcnQgKiBmcm9tICcuL3BpcGVzL3N0eWxlLWJ5cGFzcy5waXBlJztcbmV4cG9ydCAqIGZyb20gJy4vcGlwZXMvdXJsLWJ5cGFzcy5waXBlJztcblxuLyoqXG4qKiBTZXJ2aWNlc1xuKi9cbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZXMvZGVwZW5kZW5jeS5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZXMvZGlhbG9nLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcy9mZXRjaC5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZXMva2V5Ym9hcmQuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzL2ZpbGUuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzL3RoZW1lLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcy9uYXZpZ2F0aW9uLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcy9jb21tYW5kLXBhbGV0dGUuc2VydmljZSc7XG4vLyBleHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzL3NlcnZpY2V3b3JrZXIuc2VydmljZSc7XG5cbi8qKlxuKiogTGF6eSBsb2FkZXIgY29tcG9uZW50ICYgc2VydmljZVxuKi9cbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9sYXp5LWxvYWRlci9sYXp5LWxvYWRlci5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9sYXp5LWxvYWRlci9sYXp5LWxvYWRlci5tb2R1bGUnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL2xhenktbG9hZGVyL2xhenktbG9hZGVyLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvbGF6eS1sb2FkZXIvdHlwZXMnO1xuXG4vKipcbioqIER5bmFtaWMgSFRNTCAoV0lQKVxuKi9cbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9keW5hbWljLWh0bWwvZHluYW1pYy1odG1sLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL2R5bmFtaWMtaHRtbC9keW5hbWljLWh0bWwubW9kdWxlJztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9keW5hbWljLWh0bWwvZHluYW1pYy1odG1sLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvZHluYW1pYy1odG1sL3R5cGVzJztcblxuLyoqXG4gKiogQ29tcG9uZW50c1xuICovXG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvZmlsZW1hbmFnZXIvZmlsZW1hbmFnZXIuY29tcG9uZW50Jztcbi8vIGV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy9tdXNpYy1saWJyYXJ5L211c2ljLWxpYnJhcnkuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50cy90YWJ1bGF0b3IvdGFidWxhdG9yLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudHMvdnNjb2RlL3ZzY29kZS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL3BhcmFsbGF4LWNhcmQvcGFyYWxsYXgtY2FyZC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL3JlYWN0LW1hZ2ljLXdyYXBwZXIvcmVhY3QtbWFnaWMtd3JhcHBlci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jb21wb25lbnRzL3R5cGVzJztcblxuXG5leHBvcnQgeyBDb25zb2xlTG9nZ2VyLCBMb2dJY29uIH0gZnJvbSAnLi91dGlscy9pbmRleCc7XG4iXX0=
@@ -11376,6 +11376,138 @@ const InstallMonacoUMD = async (path) => {
11376
11376
  });
11377
11377
  };
11378
11378
 
11379
+ class ParallaxCardComponent {
11380
+ get wrapper() { return this.element.nativeElement; }
11381
+ get cardFront() { return this.wrapper.querySelector('.card.front'); }
11382
+ get cardBack() { return this.wrapper.querySelector('.card.backface'); }
11383
+ get backgroundElement() { return this.cardFront.querySelector('.card-bg'); }
11384
+ get backfaceBackgroundElement() { return this.cardBack.querySelector('.card-bg'); }
11385
+ constructor(element) {
11386
+ this.element = element;
11387
+ /**
11388
+ *
11389
+ */
11390
+ this.loaded = new EventEmitter();
11391
+ /**
11392
+ * Width of the card
11393
+ * @default `240px`
11394
+ */
11395
+ this.width = '240px';
11396
+ /**
11397
+ * Height of the card
11398
+ * @default `320px`
11399
+ */
11400
+ this.height = '320px';
11401
+ /**
11402
+ * Inset padding of the parallax
11403
+ * @default 80
11404
+ */
11405
+ this.bgInset = 80;
11406
+ /**
11407
+ * Duration for the flip animation in ms
11408
+ * @default 80
11409
+ */
11410
+ this.flipAnimationDuration = 1600;
11411
+ this.renderCardFront = true;
11412
+ this.renderCardBack = true;
11413
+ this.showBackOfCard = false;
11414
+ this.pointerX = 0;
11415
+ this.pointerY = 0;
11416
+ this.pointerLeave = 0;
11417
+ this.render = () => {
11418
+ const { width, height } = this.wrapper.getBoundingClientRect();
11419
+ const mousePX = this.pointerX / width;
11420
+ const mousePY = this.pointerY / height;
11421
+ // Rotation factors
11422
+ const rX = mousePX * this.bgInset / 1.75;
11423
+ const rY = mousePY * -this.bgInset / 1.75;
11424
+ // Translation factors
11425
+ const tX = mousePX * -this.bgInset * 2;
11426
+ const tY = mousePY * -this.bgInset * 2;
11427
+ if (this.renderCardFront) {
11428
+ this.backgroundElement.style.transform = `translateX(${tX}px) translateY(${tY}px)`;
11429
+ }
11430
+ if (this.renderCardBack) {
11431
+ this.backfaceBackgroundElement.style.transform = `translateX(${tX}px) translateY(${tY}px)`;
11432
+ }
11433
+ if (this.showBackOfCard) {
11434
+ this.cardFront.style.transform = `rotateY(180deg) rotateX(${-rY}deg)`;
11435
+ this.cardBack.style.transform = `rotateY(${-rX}deg) rotateX(${-rY}deg)`;
11436
+ }
11437
+ else {
11438
+ this.cardFront.style.transform = `rotateY(${rX}deg) rotateX(${rY}deg)`;
11439
+ this.cardBack.style.transform = `rotateY(180deg) rotateX(${-rY}deg)`;
11440
+ }
11441
+ };
11442
+ }
11443
+ ngAfterViewInit() {
11444
+ const el = this.wrapper;
11445
+ // Directly attach events to the wrapper
11446
+ el.onpointermove = (e) => this.onPointerMove(e);
11447
+ el.onpointerenter = () => this.onPointerEnter();
11448
+ el.onpointerleave = () => this.onPointerLeave();
11449
+ el.onclick = () => this.onClick();
11450
+ this.loaded.emit();
11451
+ }
11452
+ onPointerMove(e) {
11453
+ const { width, height, left, top } = this.wrapper.getBoundingClientRect();
11454
+ this.pointerX = e.pageX - left - (width / 2);
11455
+ this.pointerY = e.pageY - top - (height / 2);
11456
+ this.render();
11457
+ }
11458
+ onPointerEnter() {
11459
+ clearTimeout(this.pointerLeave);
11460
+ }
11461
+ onPointerLeave() {
11462
+ this.pointerLeave = setTimeout(() => {
11463
+ this.pointerX = 0;
11464
+ this.pointerY = 0;
11465
+ this.render();
11466
+ }, 600);
11467
+ }
11468
+ // TODO: This can get intercepted in some states
11469
+ onClick() {
11470
+ this.showBackOfCard = !this.showBackOfCard;
11471
+ this.render();
11472
+ }
11473
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ParallaxCardComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
11474
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.1.2", type: ParallaxCardComponent, isStandalone: true, selector: "ngx-parallax-card", inputs: { width: "width", height: "height", bgInset: "bgInset", flipAnimationDuration: "flipAnimationDuration" }, outputs: { loaded: "loaded" }, host: { properties: { "style.width": "width", "style.height": "height", "style.--card-bg-inset": "-bgInset+\"px\"", "style.--flip-animation-duration": "flipAnimationDuration+\"ms\"", "class.flip": "showBackOfCard" } }, queries: [{ propertyName: "content", first: true, predicate: ["content"], descendants: true, read: TemplateRef }, { propertyName: "background", first: true, predicate: ["background"], descendants: true, read: TemplateRef }, { propertyName: "backContent", first: true, predicate: ["backContent"], descendants: true, read: TemplateRef }, { propertyName: "backBackground", first: true, predicate: ["backBackground"], descendants: true, read: TemplateRef }], ngImport: i0, template: "@if (renderCardBack) {\n <div class=\"card backface\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"backBackground\" />\n </div>\n <div class=\"card-content\">\n <ng-template [ngTemplateOutlet]=\"backContent\" />\n </div>\n </div>\n}\n\n@if (renderCardFront) {\n <div class=\"card front\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"background\" />\n </div>\n <div class=\"card-content\">\n @if (content) {\n <ng-template [ngTemplateOutlet]=\"content\"/>\n }\n @else {\n <ng-content/>\n }\n </div>\n </div>\n}\n\n\n", styles: [":host{display:block;position:relative;cursor:pointer;--easing-function: cubic-bezier(.23, 1, .32, 1);--border-radius: 9px}.card{position:absolute;height:100%;width:100%;background-color:#333;backface-visibility:hidden;overflow:hidden;box-shadow:#0000 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px;border-radius:var(--border-radius);transition:transform var(--flip-animation-duration) var(--easing-function),box-shadow 2s var(--easing-function);transform-style:preserve-3d}.card:hover{box-shadow:#fff6 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px}.card:hover .card-content{border:2px solid #fff}.card-bg{inset:var(--card-bg-inset);position:absolute;pointer-events:none;transition:transform .6s var(--easing-function)}.card-content{position:absolute;top:0;color:#fff;height:100%;width:100%;z-index:1;border:2px solid rgba(255,255,255,0);border-radius:var(--border-radius);transition:transform .6s var(--easing-function),border-color .6s var(--easing-function)}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }] }); }
11475
+ }
11476
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ParallaxCardComponent, decorators: [{
11477
+ type: Component,
11478
+ args: [{ selector: 'ngx-parallax-card', imports: [
11479
+ NgTemplateOutlet
11480
+ ], host: {
11481
+ '[style.width]': 'width',
11482
+ '[style.height]': 'height',
11483
+ '[style.--card-bg-inset]': '-bgInset+"px"',
11484
+ '[style.--flip-animation-duration]': 'flipAnimationDuration+"ms"',
11485
+ '[class.flip]': 'showBackOfCard'
11486
+ }, standalone: true, template: "@if (renderCardBack) {\n <div class=\"card backface\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"backBackground\" />\n </div>\n <div class=\"card-content\">\n <ng-template [ngTemplateOutlet]=\"backContent\" />\n </div>\n </div>\n}\n\n@if (renderCardFront) {\n <div class=\"card front\">\n <div class=\"card-bg\" style=\"transform: translateX(0) translateY(0)\">\n <ng-template [ngTemplateOutlet]=\"background\" />\n </div>\n <div class=\"card-content\">\n @if (content) {\n <ng-template [ngTemplateOutlet]=\"content\"/>\n }\n @else {\n <ng-content/>\n }\n </div>\n </div>\n}\n\n\n", styles: [":host{display:block;position:relative;cursor:pointer;--easing-function: cubic-bezier(.23, 1, .32, 1);--border-radius: 9px}.card{position:absolute;height:100%;width:100%;background-color:#333;backface-visibility:hidden;overflow:hidden;box-shadow:#0000 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px;border-radius:var(--border-radius);transition:transform var(--flip-animation-duration) var(--easing-function),box-shadow 2s var(--easing-function);transform-style:preserve-3d}.card:hover{box-shadow:#fff6 0 0 40px 5px,#000000a8 0 30px 60px,inset #333 0 0 0 5px}.card:hover .card-content{border:2px solid #fff}.card-bg{inset:var(--card-bg-inset);position:absolute;pointer-events:none;transition:transform .6s var(--easing-function)}.card-content{position:absolute;top:0;color:#fff;height:100%;width:100%;z-index:1;border:2px solid rgba(255,255,255,0);border-radius:var(--border-radius);transition:transform .6s var(--easing-function),border-color .6s var(--easing-function)}\n"] }]
11487
+ }], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { content: [{
11488
+ type: ContentChild,
11489
+ args: ['content', { read: TemplateRef }]
11490
+ }], background: [{
11491
+ type: ContentChild,
11492
+ args: ['background', { read: TemplateRef }]
11493
+ }], backContent: [{
11494
+ type: ContentChild,
11495
+ args: ['backContent', { read: TemplateRef }]
11496
+ }], backBackground: [{
11497
+ type: ContentChild,
11498
+ args: ['backBackground', { read: TemplateRef }]
11499
+ }], loaded: [{
11500
+ type: Output
11501
+ }], width: [{
11502
+ type: Input
11503
+ }], height: [{
11504
+ type: Input
11505
+ }], bgInset: [{
11506
+ type: Input
11507
+ }], flipAnimationDuration: [{
11508
+ type: Input
11509
+ }] } });
11510
+
11379
11511
  /**
11380
11512
  * Extend this component to automatically generate
11381
11513
  * bindings to a React component.
@@ -11453,5 +11585,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImpor
11453
11585
  * Generated bundle index. Do not edit.
11454
11586
  */
11455
11587
 
11456
- export { CommandPaletteService, ComponentResolveStrategy, ConsoleLogger, DependencyService, DialogService, DynamicHTMLComponent, DynamicHTMLOptions, DynamicHTMLRenderer, Fetch, FileService, FilemanagerComponent, HtmlBypass, InstallMonacoUMD, KeyboardService, LazyLoaderComponent, LazyLoaderModule, LazyLoaderService, LogIcon, MenuDirective, NGX_DYNAMIC_CONFIG, NGX_IMAGE_CACHE_CONFIG, NGX_LAZY_LOADER_CONFIG, NGX_WEB_COMPONENTS_CONFIG, NavigationService, NgxDynamicHTMLModule, NgxImageCacheDirective, OnMount, ReactMagicWrapperComponent, ResourceBypass, ScriptBypass, StyleBypass, TabulatorComponent, ThemeService, TooltipDirective, UrlBypass, VscodeComponent, openMenu, openTooltip };
11588
+ export { CommandPaletteService, ComponentResolveStrategy, ConsoleLogger, DependencyService, DialogService, DynamicHTMLComponent, DynamicHTMLOptions, DynamicHTMLRenderer, Fetch, FileService, FilemanagerComponent, HtmlBypass, InstallMonacoUMD, KeyboardService, LazyLoaderComponent, LazyLoaderModule, LazyLoaderService, LogIcon, MenuDirective, NGX_DYNAMIC_CONFIG, NGX_IMAGE_CACHE_CONFIG, NGX_LAZY_LOADER_CONFIG, NGX_WEB_COMPONENTS_CONFIG, NavigationService, NgxDynamicHTMLModule, NgxImageCacheDirective, OnMount, ParallaxCardComponent, ReactMagicWrapperComponent, ResourceBypass, ScriptBypass, StyleBypass, TabulatorComponent, ThemeService, TooltipDirective, UrlBypass, VscodeComponent, openMenu, openTooltip };
11457
11589
  //# sourceMappingURL=dotglitch-ngx-common.mjs.map