@sitecore-jss/sitecore-jss-angular 21.2.2-canary.0 → 21.2.2
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.
- package/LICENSE.txt +202 -202
- package/README.md +7 -7
- package/angular.json +48 -48
- package/dist/README.md +7 -7
- package/dist/components/data-resolver-factory.d.ts +13 -13
- package/dist/components/date.directive.d.ts +20 -20
- package/dist/components/editframe.component.d.ts +22 -22
- package/dist/components/file.directive.d.ts +17 -17
- package/dist/components/generic-link.directive.d.ts +20 -20
- package/dist/components/guard-resolver-factory.d.ts +11 -11
- package/dist/components/hidden-rendering.component.d.ts +7 -7
- package/dist/components/image.directive.d.ts +35 -35
- package/dist/components/link.directive.d.ts +26 -26
- package/dist/components/missing-component.component.d.ts +7 -7
- package/dist/components/placeholder-loading.directive.d.ts +8 -8
- package/dist/components/placeholder.component.d.ts +59 -59
- package/dist/components/placeholder.token.d.ts +64 -64
- package/dist/components/raw.component.d.ts +12 -12
- package/dist/components/render-component.component.d.ts +33 -33
- package/dist/components/render-each.directive.d.ts +8 -8
- package/dist/components/render-empty.directive.d.ts +8 -8
- package/dist/components/rendering-field.d.ts +41 -41
- package/dist/components/rendering.d.ts +5 -5
- package/dist/components/rich-text.directive.d.ts +18 -18
- package/dist/components/router-link.directive.d.ts +19 -19
- package/dist/components/text.directive.d.ts +16 -16
- package/dist/esm2020/components/data-resolver-factory.mjs +57 -57
- package/dist/esm2020/components/date.directive.mjs +59 -59
- package/dist/esm2020/components/editframe.component.mjs +65 -65
- package/dist/esm2020/components/file.directive.mjs +44 -44
- package/dist/esm2020/components/generic-link.directive.mjs +64 -64
- package/dist/esm2020/components/guard-resolver-factory.mjs +87 -87
- package/dist/esm2020/components/hidden-rendering.component.mjs +20 -20
- package/dist/esm2020/components/image.directive.mjs +137 -137
- package/dist/esm2020/components/link.directive.mjs +117 -117
- package/dist/esm2020/components/missing-component.component.mjs +15 -15
- package/dist/esm2020/components/placeholder-loading.directive.mjs +15 -15
- package/dist/esm2020/components/placeholder.component.mjs +250 -250
- package/dist/esm2020/components/placeholder.token.mjs +23 -23
- package/dist/esm2020/components/raw.component.mjs +35 -35
- package/dist/esm2020/components/render-component.component.mjs +90 -90
- package/dist/esm2020/components/render-each.directive.mjs +15 -15
- package/dist/esm2020/components/render-empty.directive.mjs +15 -15
- package/dist/esm2020/components/rendering-field.mjs +1 -1
- package/dist/esm2020/components/rendering.mjs +7 -7
- package/dist/esm2020/components/rich-text.directive.mjs +62 -62
- package/dist/esm2020/components/router-link.directive.mjs +47 -47
- package/dist/esm2020/components/text.directive.mjs +57 -57
- package/dist/esm2020/jss-component-factory.service.mjs +83 -83
- package/dist/esm2020/lib.module.mjs +167 -167
- package/dist/esm2020/public_api.mjs +24 -24
- package/dist/esm2020/sitecore-jss-sitecore-jss-angular.mjs +4 -4
- package/dist/esm2020/utils.mjs +16 -16
- package/dist/fesm2015/sitecore-jss-sitecore-jss-angular.mjs +1415 -1415
- package/dist/fesm2020/sitecore-jss-sitecore-jss-angular.mjs +1422 -1422
- package/dist/index.d.ts +5 -5
- package/dist/jss-component-factory.service.d.ts +25 -25
- package/dist/lib.module.d.ts +46 -46
- package/dist/package.json +2 -2
- package/dist/public_api.d.ts +25 -25
- package/dist/typings/README.md +2 -2
- package/dist/utils.d.ts +7 -7
- package/ng-package.json +13 -13
- package/package.json +3 -3
- package/tsconfig.json +9 -9
- package/tsconfig.spec.json +16 -16
- package/typings/README.md +2 -2
|
@@ -1,138 +1,138 @@
|
|
|
1
|
-
import { Directive, Input, } from '@angular/core';
|
|
2
|
-
import { mediaApi } from '@sitecore-jss/sitecore-jss/media';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
export class ImageDirective {
|
|
5
|
-
constructor(viewContainer, templateRef, renderer, elementRef) {
|
|
6
|
-
this.viewContainer = viewContainer;
|
|
7
|
-
this.templateRef = templateRef;
|
|
8
|
-
this.renderer = renderer;
|
|
9
|
-
this.elementRef = elementRef;
|
|
10
|
-
this.editable = true;
|
|
11
|
-
this.urlParams = {};
|
|
12
|
-
this.attrs = {};
|
|
13
|
-
this.inlineRef = null;
|
|
14
|
-
}
|
|
15
|
-
ngOnChanges(changes) {
|
|
16
|
-
if (changes.field || changes.editable || changes.urlParams || changes.attrs) {
|
|
17
|
-
this.viewContainer.clear();
|
|
18
|
-
if (this.inlineRef) {
|
|
19
|
-
this.inlineRef.remove();
|
|
20
|
-
this.inlineRef = null;
|
|
21
|
-
}
|
|
22
|
-
this.updateView();
|
|
23
|
-
}
|
|
24
|
-
}
|
|
25
|
-
updateView() {
|
|
26
|
-
const overrideAttrs = {
|
|
27
|
-
...this.getElementAttrs(),
|
|
28
|
-
...this.attrs,
|
|
29
|
-
};
|
|
30
|
-
const media = this.field;
|
|
31
|
-
if (!media || (!media.editable && !media.value && !media.src)) {
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
34
|
-
let attrs = {};
|
|
35
|
-
// we likely have an experience editor value, should be a string
|
|
36
|
-
if (this.editable && media.editable) {
|
|
37
|
-
const foundImg = mediaApi.findEditorImageTag(media.editable);
|
|
38
|
-
if (!foundImg) {
|
|
39
|
-
return this.renderInlineWrapper(media.editable);
|
|
40
|
-
}
|
|
41
|
-
attrs = this.getImageAttrs(foundImg.attrs, overrideAttrs, this.urlParams);
|
|
42
|
-
if (!attrs) {
|
|
43
|
-
return this.renderInlineWrapper(media.editable);
|
|
44
|
-
}
|
|
45
|
-
const tempImg = this.renderer.createElement('img');
|
|
46
|
-
Object.entries(attrs).forEach(([key, attrValue]) => tempImg.setAttribute(key, attrValue));
|
|
47
|
-
const editableMarkup = media.editable.replace(foundImg.imgTag, tempImg.outerHTML);
|
|
48
|
-
return this.renderInlineWrapper(editableMarkup);
|
|
49
|
-
}
|
|
50
|
-
// some wise-guy/gal is passing in a 'raw' image object value
|
|
51
|
-
const img = media.src ? media : media.value;
|
|
52
|
-
if (!img) {
|
|
53
|
-
return null;
|
|
54
|
-
}
|
|
55
|
-
attrs = this.getImageAttrs(img, overrideAttrs, this.urlParams);
|
|
56
|
-
if (attrs) {
|
|
57
|
-
this.renderTemplate(attrs);
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
getImageAttrs(fieldAttrs, parsedAttrs, imageParams) {
|
|
61
|
-
const combinedAttrs = {
|
|
62
|
-
...fieldAttrs,
|
|
63
|
-
...parsedAttrs,
|
|
64
|
-
};
|
|
65
|
-
// eslint-disable-next-line prefer-const
|
|
66
|
-
let { src, srcSet, ...otherAttrs } = combinedAttrs;
|
|
67
|
-
if (!src) {
|
|
68
|
-
return null;
|
|
69
|
-
}
|
|
70
|
-
const newAttrs = {
|
|
71
|
-
...otherAttrs,
|
|
72
|
-
};
|
|
73
|
-
// update image URL for jss handler and image rendering params
|
|
74
|
-
src = mediaApi.updateImageUrl(src, imageParams, this.mediaUrlPrefix);
|
|
75
|
-
if (srcSet) {
|
|
76
|
-
// replace with HTML-formatted srcset, including updated image URLs
|
|
77
|
-
newAttrs.srcSet = mediaApi.getSrcSet(src, srcSet, imageParams, this.mediaUrlPrefix);
|
|
78
|
-
}
|
|
79
|
-
else {
|
|
80
|
-
newAttrs.src = src;
|
|
81
|
-
}
|
|
82
|
-
return newAttrs;
|
|
83
|
-
}
|
|
84
|
-
renderTemplate(imageProps) {
|
|
85
|
-
const viewRef = this.viewContainer.createEmbeddedView(this.templateRef);
|
|
86
|
-
viewRef.rootNodes.forEach((node) => {
|
|
87
|
-
Object.entries(imageProps).forEach(([key, imgPropVal]) => this.renderer.setAttribute(node, key, imgPropVal));
|
|
88
|
-
});
|
|
89
|
-
}
|
|
90
|
-
getElementAttrs() {
|
|
91
|
-
const view = this.templateRef.createEmbeddedView(null);
|
|
92
|
-
const element = view.rootNodes[0];
|
|
93
|
-
if (!element) {
|
|
94
|
-
view.destroy();
|
|
95
|
-
return {};
|
|
96
|
-
}
|
|
97
|
-
const attrs = {};
|
|
98
|
-
for (let i = 0; i < element.attributes.length; i++) {
|
|
99
|
-
const attr = element.attributes.item(i);
|
|
100
|
-
if (attr) {
|
|
101
|
-
attrs[attr.name] = attr.value;
|
|
102
|
-
}
|
|
103
|
-
}
|
|
104
|
-
view.destroy();
|
|
105
|
-
return attrs;
|
|
106
|
-
}
|
|
107
|
-
renderInlineWrapper(editable) {
|
|
108
|
-
const span = this.renderer.createElement('span');
|
|
109
|
-
span.className = 'sc-image-wrapper';
|
|
110
|
-
span.innerHTML = editable;
|
|
111
|
-
const parentNode = this.renderer.parentNode(this.elementRef.nativeElement);
|
|
112
|
-
this.renderer.insertBefore(parentNode, span, this.elementRef.nativeElement);
|
|
113
|
-
parentNode.removeChild(this.elementRef.nativeElement);
|
|
114
|
-
this.inlineRef = span;
|
|
115
|
-
}
|
|
116
|
-
}
|
|
117
|
-
ImageDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ImageDirective, deps: [{ token: i0.ViewContainerRef }, { token: i0.TemplateRef }, { token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
118
|
-
ImageDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.2.12", type: ImageDirective, selector: "[scImage]", inputs: { field: ["scImage", "field"], editable: ["scImageEditable", "editable"], mediaUrlPrefix: ["scImageMediaUrlPrefix", "mediaUrlPrefix"], urlParams: ["scImageUrlParams", "urlParams"], attrs: ["scImageAttrs", "attrs"] }, usesOnChanges: true, ngImport: i0 });
|
|
119
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ImageDirective, decorators: [{
|
|
120
|
-
type: Directive,
|
|
121
|
-
args: [{ selector: '[scImage]' }]
|
|
122
|
-
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.TemplateRef }, { type: i0.Renderer2 }, { type: i0.ElementRef }]; }, propDecorators: { field: [{
|
|
123
|
-
type: Input,
|
|
124
|
-
args: ['scImage']
|
|
125
|
-
}], editable: [{
|
|
126
|
-
type: Input,
|
|
127
|
-
args: ['scImageEditable']
|
|
128
|
-
}], mediaUrlPrefix: [{
|
|
129
|
-
type: Input,
|
|
130
|
-
args: ['scImageMediaUrlPrefix']
|
|
131
|
-
}], urlParams: [{
|
|
132
|
-
type: Input,
|
|
133
|
-
args: ['scImageUrlParams']
|
|
134
|
-
}], attrs: [{
|
|
135
|
-
type: Input,
|
|
136
|
-
args: ['scImageAttrs']
|
|
137
|
-
}] } });
|
|
1
|
+
import { Directive, Input, } from '@angular/core';
|
|
2
|
+
import { mediaApi } from '@sitecore-jss/sitecore-jss/media';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class ImageDirective {
|
|
5
|
+
constructor(viewContainer, templateRef, renderer, elementRef) {
|
|
6
|
+
this.viewContainer = viewContainer;
|
|
7
|
+
this.templateRef = templateRef;
|
|
8
|
+
this.renderer = renderer;
|
|
9
|
+
this.elementRef = elementRef;
|
|
10
|
+
this.editable = true;
|
|
11
|
+
this.urlParams = {};
|
|
12
|
+
this.attrs = {};
|
|
13
|
+
this.inlineRef = null;
|
|
14
|
+
}
|
|
15
|
+
ngOnChanges(changes) {
|
|
16
|
+
if (changes.field || changes.editable || changes.urlParams || changes.attrs) {
|
|
17
|
+
this.viewContainer.clear();
|
|
18
|
+
if (this.inlineRef) {
|
|
19
|
+
this.inlineRef.remove();
|
|
20
|
+
this.inlineRef = null;
|
|
21
|
+
}
|
|
22
|
+
this.updateView();
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
updateView() {
|
|
26
|
+
const overrideAttrs = {
|
|
27
|
+
...this.getElementAttrs(),
|
|
28
|
+
...this.attrs,
|
|
29
|
+
};
|
|
30
|
+
const media = this.field;
|
|
31
|
+
if (!media || (!media.editable && !media.value && !media.src)) {
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
let attrs = {};
|
|
35
|
+
// we likely have an experience editor value, should be a string
|
|
36
|
+
if (this.editable && media.editable) {
|
|
37
|
+
const foundImg = mediaApi.findEditorImageTag(media.editable);
|
|
38
|
+
if (!foundImg) {
|
|
39
|
+
return this.renderInlineWrapper(media.editable);
|
|
40
|
+
}
|
|
41
|
+
attrs = this.getImageAttrs(foundImg.attrs, overrideAttrs, this.urlParams);
|
|
42
|
+
if (!attrs) {
|
|
43
|
+
return this.renderInlineWrapper(media.editable);
|
|
44
|
+
}
|
|
45
|
+
const tempImg = this.renderer.createElement('img');
|
|
46
|
+
Object.entries(attrs).forEach(([key, attrValue]) => tempImg.setAttribute(key, attrValue));
|
|
47
|
+
const editableMarkup = media.editable.replace(foundImg.imgTag, tempImg.outerHTML);
|
|
48
|
+
return this.renderInlineWrapper(editableMarkup);
|
|
49
|
+
}
|
|
50
|
+
// some wise-guy/gal is passing in a 'raw' image object value
|
|
51
|
+
const img = media.src ? media : media.value;
|
|
52
|
+
if (!img) {
|
|
53
|
+
return null;
|
|
54
|
+
}
|
|
55
|
+
attrs = this.getImageAttrs(img, overrideAttrs, this.urlParams);
|
|
56
|
+
if (attrs) {
|
|
57
|
+
this.renderTemplate(attrs);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
getImageAttrs(fieldAttrs, parsedAttrs, imageParams) {
|
|
61
|
+
const combinedAttrs = {
|
|
62
|
+
...fieldAttrs,
|
|
63
|
+
...parsedAttrs,
|
|
64
|
+
};
|
|
65
|
+
// eslint-disable-next-line prefer-const
|
|
66
|
+
let { src, srcSet, ...otherAttrs } = combinedAttrs;
|
|
67
|
+
if (!src) {
|
|
68
|
+
return null;
|
|
69
|
+
}
|
|
70
|
+
const newAttrs = {
|
|
71
|
+
...otherAttrs,
|
|
72
|
+
};
|
|
73
|
+
// update image URL for jss handler and image rendering params
|
|
74
|
+
src = mediaApi.updateImageUrl(src, imageParams, this.mediaUrlPrefix);
|
|
75
|
+
if (srcSet) {
|
|
76
|
+
// replace with HTML-formatted srcset, including updated image URLs
|
|
77
|
+
newAttrs.srcSet = mediaApi.getSrcSet(src, srcSet, imageParams, this.mediaUrlPrefix);
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
newAttrs.src = src;
|
|
81
|
+
}
|
|
82
|
+
return newAttrs;
|
|
83
|
+
}
|
|
84
|
+
renderTemplate(imageProps) {
|
|
85
|
+
const viewRef = this.viewContainer.createEmbeddedView(this.templateRef);
|
|
86
|
+
viewRef.rootNodes.forEach((node) => {
|
|
87
|
+
Object.entries(imageProps).forEach(([key, imgPropVal]) => this.renderer.setAttribute(node, key, imgPropVal));
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
getElementAttrs() {
|
|
91
|
+
const view = this.templateRef.createEmbeddedView(null);
|
|
92
|
+
const element = view.rootNodes[0];
|
|
93
|
+
if (!element) {
|
|
94
|
+
view.destroy();
|
|
95
|
+
return {};
|
|
96
|
+
}
|
|
97
|
+
const attrs = {};
|
|
98
|
+
for (let i = 0; i < element.attributes.length; i++) {
|
|
99
|
+
const attr = element.attributes.item(i);
|
|
100
|
+
if (attr) {
|
|
101
|
+
attrs[attr.name] = attr.value;
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
view.destroy();
|
|
105
|
+
return attrs;
|
|
106
|
+
}
|
|
107
|
+
renderInlineWrapper(editable) {
|
|
108
|
+
const span = this.renderer.createElement('span');
|
|
109
|
+
span.className = 'sc-image-wrapper';
|
|
110
|
+
span.innerHTML = editable;
|
|
111
|
+
const parentNode = this.renderer.parentNode(this.elementRef.nativeElement);
|
|
112
|
+
this.renderer.insertBefore(parentNode, span, this.elementRef.nativeElement);
|
|
113
|
+
parentNode.removeChild(this.elementRef.nativeElement);
|
|
114
|
+
this.inlineRef = span;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
ImageDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ImageDirective, deps: [{ token: i0.ViewContainerRef }, { token: i0.TemplateRef }, { token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
118
|
+
ImageDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.2.12", type: ImageDirective, selector: "[scImage]", inputs: { field: ["scImage", "field"], editable: ["scImageEditable", "editable"], mediaUrlPrefix: ["scImageMediaUrlPrefix", "mediaUrlPrefix"], urlParams: ["scImageUrlParams", "urlParams"], attrs: ["scImageAttrs", "attrs"] }, usesOnChanges: true, ngImport: i0 });
|
|
119
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ImageDirective, decorators: [{
|
|
120
|
+
type: Directive,
|
|
121
|
+
args: [{ selector: '[scImage]' }]
|
|
122
|
+
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.TemplateRef }, { type: i0.Renderer2 }, { type: i0.ElementRef }]; }, propDecorators: { field: [{
|
|
123
|
+
type: Input,
|
|
124
|
+
args: ['scImage']
|
|
125
|
+
}], editable: [{
|
|
126
|
+
type: Input,
|
|
127
|
+
args: ['scImageEditable']
|
|
128
|
+
}], mediaUrlPrefix: [{
|
|
129
|
+
type: Input,
|
|
130
|
+
args: ['scImageMediaUrlPrefix']
|
|
131
|
+
}], urlParams: [{
|
|
132
|
+
type: Input,
|
|
133
|
+
args: ['scImageUrlParams']
|
|
134
|
+
}], attrs: [{
|
|
135
|
+
type: Input,
|
|
136
|
+
args: ['scImageAttrs']
|
|
137
|
+
}] } });
|
|
138
138
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image.directive.js","sourceRoot":"","sources":["../../../src/components/image.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Directive,\r\n  ElementRef,\r\n  Input,\r\n  OnChanges,\r\n  Renderer2,\r\n  SimpleChanges,\r\n  TemplateRef,\r\n  ViewContainerRef,\r\n} from '@angular/core';\r\nimport { mediaApi } from '@sitecore-jss/sitecore-jss/media';\r\nimport { ImageField, ImageFieldValue } from './rendering-field';\r\n\r\n@Directive({ selector: '[scImage]' })\r\nexport class ImageDirective implements OnChanges {\r\n  @Input('scImage') field: ImageField | '';\r\n\r\n  @Input('scImageEditable') editable = true;\r\n\r\n  /**\r\n   * Custom regexp that finds media URL prefix that will be replaced by `/-/jssmedia` or `/~/jssmedia`.\r\n   * @example\r\n   * /\\/([-~]{1})assets\\//i\r\n   * /-assets/website -> /-/jssmedia/website\r\n   * /~assets/website -> /~/jssmedia/website\r\n   */\r\n  @Input('scImageMediaUrlPrefix') mediaUrlPrefix?: RegExp;\r\n\r\n  @Input('scImageUrlParams') urlParams: { [param: string]: string | number } = {};\r\n\r\n  @Input('scImageAttrs') attrs: { [param: string]: unknown } = {};\r\n\r\n  private inlineRef: HTMLSpanElement | null = null;\r\n\r\n  constructor(\r\n    private viewContainer: ViewContainerRef,\r\n    private templateRef: TemplateRef<unknown>,\r\n    private renderer: Renderer2,\r\n    private elementRef: ElementRef\r\n  ) {}\r\n\r\n  ngOnChanges(changes: SimpleChanges) {\r\n    if (changes.field || changes.editable || changes.urlParams || changes.attrs) {\r\n      this.viewContainer.clear();\r\n      if (this.inlineRef) {\r\n        this.inlineRef.remove();\r\n        this.inlineRef = null;\r\n      }\r\n\r\n      this.updateView();\r\n    }\r\n  }\r\n\r\n  private updateView() {\r\n    const overrideAttrs = {\r\n      ...this.getElementAttrs(),\r\n      ...this.attrs,\r\n    };\r\n    const media = this.field;\r\n\r\n    if (!media || (!media.editable && !media.value && !media.src)) {\r\n      return;\r\n    }\r\n\r\n    let attrs: { [attr: string]: string } | null = {};\r\n\r\n    // we likely have an experience editor value, should be a string\r\n    if (this.editable && media.editable) {\r\n      const foundImg = mediaApi.findEditorImageTag(media.editable);\r\n      if (!foundImg) {\r\n        return this.renderInlineWrapper(media.editable);\r\n      }\r\n      attrs = this.getImageAttrs(foundImg.attrs, overrideAttrs, this.urlParams);\r\n      if (!attrs) {\r\n        return this.renderInlineWrapper(media.editable);\r\n      }\r\n      const tempImg: HTMLImageElement = this.renderer.createElement('img');\r\n      Object.entries(attrs).forEach(([key, attrValue]: [string, string]) =>\r\n        tempImg.setAttribute(key, attrValue)\r\n      );\r\n      const editableMarkup = media.editable.replace(foundImg.imgTag, tempImg.outerHTML);\r\n      return this.renderInlineWrapper(editableMarkup);\r\n    }\r\n\r\n    // some wise-guy/gal is passing in a 'raw' image object value\r\n    const img = media.src ? media : media.value;\r\n    if (!img) {\r\n      return null;\r\n    }\r\n\r\n    attrs = this.getImageAttrs(img, overrideAttrs, this.urlParams);\r\n    if (attrs) {\r\n      this.renderTemplate(attrs);\r\n    }\r\n  }\r\n\r\n  private getImageAttrs(\r\n    fieldAttrs: ImageFieldValue,\r\n    parsedAttrs: { [attr: string]: unknown },\r\n    imageParams: { [param: string]: string | number }\r\n  ): { [attr: string]: string } | null {\r\n    const combinedAttrs = {\r\n      ...fieldAttrs,\r\n      ...parsedAttrs,\r\n    };\r\n    // eslint-disable-next-line prefer-const\r\n    let { src, srcSet, ...otherAttrs } = combinedAttrs;\r\n    if (!src) {\r\n      return null;\r\n    }\r\n    const newAttrs: { [attr: string]: string } = {\r\n      ...(otherAttrs as { [key: string]: string }),\r\n    };\r\n    // update image URL for jss handler and image rendering params\r\n    src = mediaApi.updateImageUrl(src, imageParams, this.mediaUrlPrefix);\r\n    if (srcSet) {\r\n      // replace with HTML-formatted srcset, including updated image URLs\r\n      newAttrs.srcSet = mediaApi.getSrcSet(src, srcSet, imageParams, this.mediaUrlPrefix);\r\n    } else {\r\n      newAttrs.src = src;\r\n    }\r\n    return newAttrs;\r\n  }\r\n\r\n  private renderTemplate(imageProps: { [prop: string]: string }) {\r\n    const viewRef = this.viewContainer.createEmbeddedView(this.templateRef);\r\n    viewRef.rootNodes.forEach((node) => {\r\n      Object.entries(imageProps).forEach(([key, imgPropVal]: [string, string]) =>\r\n        this.renderer.setAttribute(node, key, imgPropVal)\r\n      );\r\n    });\r\n  }\r\n\r\n  private getElementAttrs(): { [key: string]: string } {\r\n    const view = this.templateRef.createEmbeddedView(null);\r\n    const element: Element = view.rootNodes[0];\r\n    if (!element) {\r\n      view.destroy();\r\n      return {};\r\n    }\r\n    const attrs: { [key: string]: string } = {};\r\n    for (let i = 0; i < element.attributes.length; i++) {\r\n      const attr = element.attributes.item(i);\r\n      if (attr) {\r\n        attrs[attr.name] = attr.value;\r\n      }\r\n    }\r\n    view.destroy();\r\n    return attrs;\r\n  }\r\n\r\n  private renderInlineWrapper(editable: string) {\r\n    const span: HTMLSpanElement = this.renderer.createElement('span');\r\n    span.className = 'sc-image-wrapper';\r\n    span.innerHTML = editable;\r\n\r\n    const parentNode = this.renderer.parentNode(this.elementRef.nativeElement);\r\n    this.renderer.insertBefore(parentNode, span, this.elementRef.nativeElement);\r\n    parentNode.removeChild(this.elementRef.nativeElement);\r\n\r\n    this.inlineRef = span;\r\n  }\r\n}\r\n"]}
|
|
@@ -1,118 +1,118 @@
|
|
|
1
|
-
import { Directive, Input, } from '@angular/core';
|
|
2
|
-
import * as i0 from "@angular/core";
|
|
3
|
-
export class LinkDirective {
|
|
4
|
-
constructor(viewContainer, templateRef, renderer, elementRef) {
|
|
5
|
-
this.viewContainer = viewContainer;
|
|
6
|
-
this.templateRef = templateRef;
|
|
7
|
-
this.renderer = renderer;
|
|
8
|
-
this.elementRef = elementRef;
|
|
9
|
-
this.editable = true;
|
|
10
|
-
this.attrs = {};
|
|
11
|
-
this.inlineRef = null;
|
|
12
|
-
}
|
|
13
|
-
ngOnChanges(changes) {
|
|
14
|
-
if (changes.field || changes.editable || changes.attrs) {
|
|
15
|
-
this.viewContainer.clear();
|
|
16
|
-
if (this.inlineRef) {
|
|
17
|
-
this.inlineRef.remove();
|
|
18
|
-
this.inlineRef = null;
|
|
19
|
-
}
|
|
20
|
-
this.updateView();
|
|
21
|
-
}
|
|
22
|
-
}
|
|
23
|
-
renderTemplate(props, linkText) {
|
|
24
|
-
const viewRef = this.viewContainer.createEmbeddedView(this.templateRef);
|
|
25
|
-
viewRef.rootNodes.forEach((node) => {
|
|
26
|
-
Object.entries(props).forEach(([key, propValue]) => {
|
|
27
|
-
this.updateAttribute(node, key, propValue);
|
|
28
|
-
});
|
|
29
|
-
if (node.childNodes && node.childNodes.length === 0 && linkText) {
|
|
30
|
-
node.textContent = linkText;
|
|
31
|
-
}
|
|
32
|
-
});
|
|
33
|
-
}
|
|
34
|
-
updateAttribute(node, key, propValue) {
|
|
35
|
-
if (typeof propValue !== 'string' || !propValue || propValue === '') {
|
|
36
|
-
return;
|
|
37
|
-
}
|
|
38
|
-
if (key === 'href') {
|
|
39
|
-
const isInvalidLink = !propValue || /^https?:\/\/$/.test(propValue);
|
|
40
|
-
if (isInvalidLink) {
|
|
41
|
-
if (!node.href) {
|
|
42
|
-
return;
|
|
43
|
-
}
|
|
44
|
-
propValue = node.href;
|
|
45
|
-
}
|
|
46
|
-
this.renderer.setAttribute(node, key, propValue);
|
|
47
|
-
}
|
|
48
|
-
else if (key === 'class' && node.className !== '') {
|
|
49
|
-
this.renderer.setAttribute(node, key, `${node.className} ${propValue}`);
|
|
50
|
-
}
|
|
51
|
-
else {
|
|
52
|
-
this.renderer.setAttribute(node, key, propValue);
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
updateView() {
|
|
56
|
-
const field = this.field;
|
|
57
|
-
if (this.editable && field && field.editableFirstPart && field.editableLastPart) {
|
|
58
|
-
this.renderInlineWrapper(field.editableFirstPart, field.editableLastPart);
|
|
59
|
-
}
|
|
60
|
-
else if (field && (field.href || field.value)) {
|
|
61
|
-
const props = field.href ? field : field.value;
|
|
62
|
-
const linkText = field.text || field.value?.text || field.href || field.value?.href;
|
|
63
|
-
const anchor = props?.anchor ? `#${props.anchor}` : '';
|
|
64
|
-
const href = `${props?.href}${anchor}`;
|
|
65
|
-
const mergedAttrs = { ...props, ...this.attrs, href };
|
|
66
|
-
delete mergedAttrs.anchor;
|
|
67
|
-
this.renderTemplate(mergedAttrs, linkText);
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
renderInlineWrapper(editableFirstPart, editableLastPart) {
|
|
71
|
-
const span = this.renderer.createElement('span');
|
|
72
|
-
span.className = 'sc-link-wrapper';
|
|
73
|
-
span.innerHTML = editableFirstPart + editableLastPart;
|
|
74
|
-
// assign attributes from template to inline wrapper
|
|
75
|
-
const attrs = {
|
|
76
|
-
...this.getElementAttrs(),
|
|
77
|
-
...this.attrs,
|
|
78
|
-
};
|
|
79
|
-
Object.entries(attrs).forEach(([key, attrValue]) => this.updateAttribute(span, key, attrValue));
|
|
80
|
-
this.viewContainer.createEmbeddedView(this.templateRef);
|
|
81
|
-
const parentNode = this.renderer.parentNode(this.elementRef.nativeElement);
|
|
82
|
-
this.renderer.insertBefore(parentNode, span, this.elementRef.nativeElement);
|
|
83
|
-
this.inlineRef = span;
|
|
84
|
-
}
|
|
85
|
-
getElementAttrs() {
|
|
86
|
-
const view = this.templateRef.createEmbeddedView(null);
|
|
87
|
-
const element = view.rootNodes[0];
|
|
88
|
-
if (!element) {
|
|
89
|
-
view.destroy();
|
|
90
|
-
return {};
|
|
91
|
-
}
|
|
92
|
-
const attrs = {};
|
|
93
|
-
for (let i = 0; i < element.attributes.length; i++) {
|
|
94
|
-
const attr = element.attributes.item(i);
|
|
95
|
-
if (attr) {
|
|
96
|
-
attrs[attr.name] = attr.value;
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
view.destroy();
|
|
100
|
-
return attrs;
|
|
101
|
-
}
|
|
102
|
-
}
|
|
103
|
-
LinkDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: LinkDirective, deps: [{ token: i0.ViewContainerRef }, { token: i0.TemplateRef }, { token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
104
|
-
LinkDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.2.12", type: LinkDirective, selector: "[scLink]", inputs: { editable: ["scLinkEditable", "editable"], attrs: ["scLinkAttrs", "attrs"], field: ["scLink", "field"] }, usesOnChanges: true, ngImport: i0 });
|
|
105
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: LinkDirective, decorators: [{
|
|
106
|
-
type: Directive,
|
|
107
|
-
args: [{ selector: '[scLink]' }]
|
|
108
|
-
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.TemplateRef }, { type: i0.Renderer2 }, { type: i0.ElementRef }]; }, propDecorators: { editable: [{
|
|
109
|
-
type: Input,
|
|
110
|
-
args: ['scLinkEditable']
|
|
111
|
-
}], attrs: [{
|
|
112
|
-
type: Input,
|
|
113
|
-
args: ['scLinkAttrs']
|
|
114
|
-
}], field: [{
|
|
115
|
-
type: Input,
|
|
116
|
-
args: ['scLink']
|
|
117
|
-
}] } });
|
|
1
|
+
import { Directive, Input, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class LinkDirective {
|
|
4
|
+
constructor(viewContainer, templateRef, renderer, elementRef) {
|
|
5
|
+
this.viewContainer = viewContainer;
|
|
6
|
+
this.templateRef = templateRef;
|
|
7
|
+
this.renderer = renderer;
|
|
8
|
+
this.elementRef = elementRef;
|
|
9
|
+
this.editable = true;
|
|
10
|
+
this.attrs = {};
|
|
11
|
+
this.inlineRef = null;
|
|
12
|
+
}
|
|
13
|
+
ngOnChanges(changes) {
|
|
14
|
+
if (changes.field || changes.editable || changes.attrs) {
|
|
15
|
+
this.viewContainer.clear();
|
|
16
|
+
if (this.inlineRef) {
|
|
17
|
+
this.inlineRef.remove();
|
|
18
|
+
this.inlineRef = null;
|
|
19
|
+
}
|
|
20
|
+
this.updateView();
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
renderTemplate(props, linkText) {
|
|
24
|
+
const viewRef = this.viewContainer.createEmbeddedView(this.templateRef);
|
|
25
|
+
viewRef.rootNodes.forEach((node) => {
|
|
26
|
+
Object.entries(props).forEach(([key, propValue]) => {
|
|
27
|
+
this.updateAttribute(node, key, propValue);
|
|
28
|
+
});
|
|
29
|
+
if (node.childNodes && node.childNodes.length === 0 && linkText) {
|
|
30
|
+
node.textContent = linkText;
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
updateAttribute(node, key, propValue) {
|
|
35
|
+
if (typeof propValue !== 'string' || !propValue || propValue === '') {
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
if (key === 'href') {
|
|
39
|
+
const isInvalidLink = !propValue || /^https?:\/\/$/.test(propValue);
|
|
40
|
+
if (isInvalidLink) {
|
|
41
|
+
if (!node.href) {
|
|
42
|
+
return;
|
|
43
|
+
}
|
|
44
|
+
propValue = node.href;
|
|
45
|
+
}
|
|
46
|
+
this.renderer.setAttribute(node, key, propValue);
|
|
47
|
+
}
|
|
48
|
+
else if (key === 'class' && node.className !== '') {
|
|
49
|
+
this.renderer.setAttribute(node, key, `${node.className} ${propValue}`);
|
|
50
|
+
}
|
|
51
|
+
else {
|
|
52
|
+
this.renderer.setAttribute(node, key, propValue);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
updateView() {
|
|
56
|
+
const field = this.field;
|
|
57
|
+
if (this.editable && field && field.editableFirstPart && field.editableLastPart) {
|
|
58
|
+
this.renderInlineWrapper(field.editableFirstPart, field.editableLastPart);
|
|
59
|
+
}
|
|
60
|
+
else if (field && (field.href || field.value)) {
|
|
61
|
+
const props = field.href ? field : field.value;
|
|
62
|
+
const linkText = field.text || field.value?.text || field.href || field.value?.href;
|
|
63
|
+
const anchor = props?.anchor ? `#${props.anchor}` : '';
|
|
64
|
+
const href = `${props?.href}${anchor}`;
|
|
65
|
+
const mergedAttrs = { ...props, ...this.attrs, href };
|
|
66
|
+
delete mergedAttrs.anchor;
|
|
67
|
+
this.renderTemplate(mergedAttrs, linkText);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
renderInlineWrapper(editableFirstPart, editableLastPart) {
|
|
71
|
+
const span = this.renderer.createElement('span');
|
|
72
|
+
span.className = 'sc-link-wrapper';
|
|
73
|
+
span.innerHTML = editableFirstPart + editableLastPart;
|
|
74
|
+
// assign attributes from template to inline wrapper
|
|
75
|
+
const attrs = {
|
|
76
|
+
...this.getElementAttrs(),
|
|
77
|
+
...this.attrs,
|
|
78
|
+
};
|
|
79
|
+
Object.entries(attrs).forEach(([key, attrValue]) => this.updateAttribute(span, key, attrValue));
|
|
80
|
+
this.viewContainer.createEmbeddedView(this.templateRef);
|
|
81
|
+
const parentNode = this.renderer.parentNode(this.elementRef.nativeElement);
|
|
82
|
+
this.renderer.insertBefore(parentNode, span, this.elementRef.nativeElement);
|
|
83
|
+
this.inlineRef = span;
|
|
84
|
+
}
|
|
85
|
+
getElementAttrs() {
|
|
86
|
+
const view = this.templateRef.createEmbeddedView(null);
|
|
87
|
+
const element = view.rootNodes[0];
|
|
88
|
+
if (!element) {
|
|
89
|
+
view.destroy();
|
|
90
|
+
return {};
|
|
91
|
+
}
|
|
92
|
+
const attrs = {};
|
|
93
|
+
for (let i = 0; i < element.attributes.length; i++) {
|
|
94
|
+
const attr = element.attributes.item(i);
|
|
95
|
+
if (attr) {
|
|
96
|
+
attrs[attr.name] = attr.value;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
view.destroy();
|
|
100
|
+
return attrs;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
LinkDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: LinkDirective, deps: [{ token: i0.ViewContainerRef }, { token: i0.TemplateRef }, { token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
104
|
+
LinkDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.2.12", type: LinkDirective, selector: "[scLink]", inputs: { editable: ["scLinkEditable", "editable"], attrs: ["scLinkAttrs", "attrs"], field: ["scLink", "field"] }, usesOnChanges: true, ngImport: i0 });
|
|
105
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: LinkDirective, decorators: [{
|
|
106
|
+
type: Directive,
|
|
107
|
+
args: [{ selector: '[scLink]' }]
|
|
108
|
+
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.TemplateRef }, { type: i0.Renderer2 }, { type: i0.ElementRef }]; }, propDecorators: { editable: [{
|
|
109
|
+
type: Input,
|
|
110
|
+
args: ['scLinkEditable']
|
|
111
|
+
}], attrs: [{
|
|
112
|
+
type: Input,
|
|
113
|
+
args: ['scLinkAttrs']
|
|
114
|
+
}], field: [{
|
|
115
|
+
type: Input,
|
|
116
|
+
args: ['scLink']
|
|
117
|
+
}] } });
|
|
118
118
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link.directive.js","sourceRoot":"","sources":["../../../src/components/link.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Directive,\r\n  ElementRef,\r\n  Input,\r\n  OnChanges,\r\n  Renderer2,\r\n  SimpleChanges,\r\n  TemplateRef,\r\n  ViewContainerRef,\r\n} from '@angular/core';\r\nimport { LinkField } from './rendering-field';\r\n\r\n@Directive({ selector: '[scLink]' })\r\nexport class LinkDirective implements OnChanges {\r\n  @Input('scLinkEditable') editable = true;\r\n\r\n  @Input('scLinkAttrs') attrs: { [attr: string]: string } = {};\r\n\r\n  @Input('scLink') field: LinkField;\r\n\r\n  private inlineRef: HTMLSpanElement | null = null;\r\n\r\n  constructor(\r\n    protected viewContainer: ViewContainerRef,\r\n    protected templateRef: TemplateRef<unknown>,\r\n    protected renderer: Renderer2,\r\n    private elementRef: ElementRef\r\n  ) {}\r\n\r\n  ngOnChanges(changes: SimpleChanges) {\r\n    if (changes.field || changes.editable || changes.attrs) {\r\n      this.viewContainer.clear();\r\n      if (this.inlineRef) {\r\n        this.inlineRef.remove();\r\n        this.inlineRef = null;\r\n      }\r\n\r\n      this.updateView();\r\n    }\r\n  }\r\n\r\n  protected renderTemplate(props: { [prop: string]: unknown }, linkText?: string) {\r\n    const viewRef = this.viewContainer.createEmbeddedView(this.templateRef);\r\n\r\n    viewRef.rootNodes.forEach((node) => {\r\n      Object.entries(props).forEach(([key, propValue]) => {\r\n        this.updateAttribute(node, key, propValue);\r\n      });\r\n\r\n      if (node.childNodes && node.childNodes.length === 0 && linkText) {\r\n        node.textContent = linkText;\r\n      }\r\n    });\r\n  }\r\n\r\n  protected updateAttribute(node: HTMLElement, key: string, propValue?: unknown) {\r\n    if (typeof propValue !== 'string' || !propValue || propValue === '') {\r\n      return;\r\n    }\r\n\r\n    if (key === 'href') {\r\n      const isInvalidLink = !propValue || /^https?:\\/\\/$/.test(propValue);\r\n\r\n      if (isInvalidLink) {\r\n        if (!(node as HTMLLinkElement).href) {\r\n          return;\r\n        }\r\n\r\n        propValue = (node as HTMLLinkElement).href as string;\r\n      }\r\n      this.renderer.setAttribute(node, key, propValue as string);\r\n    } else if (key === 'class' && node.className !== '') {\r\n      this.renderer.setAttribute(node, key, `${node.className} ${propValue}`);\r\n    } else {\r\n      this.renderer.setAttribute(node, key, propValue);\r\n    }\r\n  }\r\n\r\n  private updateView() {\r\n    const field = this.field;\r\n    if (this.editable && field && field.editableFirstPart && field.editableLastPart) {\r\n      this.renderInlineWrapper(field.editableFirstPart, field.editableLastPart);\r\n    } else if (field && (field.href || field.value)) {\r\n      const props = field.href ? field : field.value;\r\n\r\n      const linkText = field.text || field.value?.text || field.href || field.value?.href;\r\n      const anchor = props?.anchor ? `#${props.anchor}` : '';\r\n      const href = `${props?.href}${anchor}`;\r\n\r\n      const mergedAttrs = { ...props, ...this.attrs, href };\r\n\r\n      delete mergedAttrs.anchor;\r\n\r\n      this.renderTemplate(mergedAttrs, linkText);\r\n    }\r\n  }\r\n\r\n  private renderInlineWrapper(editableFirstPart: string, editableLastPart: string) {\r\n    const span: HTMLSpanElement = this.renderer.createElement('span');\r\n    span.className = 'sc-link-wrapper';\r\n    span.innerHTML = editableFirstPart + editableLastPart;\r\n\r\n    // assign attributes from template to inline wrapper\r\n    const attrs = {\r\n      ...this.getElementAttrs(),\r\n      ...this.attrs,\r\n    };\r\n    Object.entries(attrs).forEach(([key, attrValue]) => this.updateAttribute(span, key, attrValue));\r\n\r\n    this.viewContainer.createEmbeddedView(this.templateRef);\r\n\r\n    const parentNode = this.renderer.parentNode(this.elementRef.nativeElement);\r\n    this.renderer.insertBefore(parentNode, span, this.elementRef.nativeElement);\r\n\r\n    this.inlineRef = span;\r\n  }\r\n\r\n  private getElementAttrs() {\r\n    const view = this.templateRef.createEmbeddedView(null);\r\n    const element: Element = view.rootNodes[0];\r\n    if (!element) {\r\n      view.destroy();\r\n      return {};\r\n    }\r\n    const attrs: { [key: string]: string } = {};\r\n    for (let i = 0; i < element.attributes.length; i++) {\r\n      const attr = element.attributes.item(i);\r\n      if (attr) {\r\n        attrs[attr.name] = attr.value;\r\n      }\r\n    }\r\n    view.destroy();\r\n    return attrs;\r\n  }\r\n}\r\n"]}
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import { Component, Input } from '@angular/core';
|
|
2
|
-
import * as i0 from "@angular/core";
|
|
3
|
-
export class MissingComponentComponent {
|
|
4
|
-
}
|
|
5
|
-
MissingComponentComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: MissingComponentComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class MissingComponentComponent {
|
|
4
|
+
}
|
|
5
|
+
MissingComponentComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: MissingComponentComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
6
6
|
MissingComponentComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: MissingComponentComponent, selector: "sc-missing-component", inputs: { rendering: "rendering" }, ngImport: i0, template: `
|
|
7
7
|
<div
|
|
8
8
|
style="background: darkorange; outline: 5px solid orange; padding: 10px; color: white; max-width: 500px;"
|
|
@@ -10,11 +10,11 @@ MissingComponentComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.
|
|
|
10
10
|
<h2>{{ rendering.componentName }}</h2>
|
|
11
11
|
<p>JSS component is missing Angular component implementation.</p>
|
|
12
12
|
</div>
|
|
13
|
-
`, isInline: true });
|
|
14
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: MissingComponentComponent, decorators: [{
|
|
15
|
-
type: Component,
|
|
16
|
-
args: [{
|
|
17
|
-
selector: 'sc-missing-component',
|
|
13
|
+
`, isInline: true });
|
|
14
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: MissingComponentComponent, decorators: [{
|
|
15
|
+
type: Component,
|
|
16
|
+
args: [{
|
|
17
|
+
selector: 'sc-missing-component',
|
|
18
18
|
template: `
|
|
19
19
|
<div
|
|
20
20
|
style="background: darkorange; outline: 5px solid orange; padding: 10px; color: white; max-width: 500px;"
|
|
@@ -22,9 +22,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
22
22
|
<h2>{{ rendering.componentName }}</h2>
|
|
23
23
|
<p>JSS component is missing Angular component implementation.</p>
|
|
24
24
|
</div>
|
|
25
|
-
`,
|
|
26
|
-
}]
|
|
27
|
-
}], propDecorators: { rendering: [{
|
|
28
|
-
type: Input
|
|
29
|
-
}] } });
|
|
25
|
+
`,
|
|
26
|
+
}]
|
|
27
|
+
}], propDecorators: { rendering: [{
|
|
28
|
+
type: Input
|
|
29
|
+
}] } });
|
|
30
30
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWlzc2luZy1jb21wb25lbnQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvbWlzc2luZy1jb21wb25lbnQuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQWNqRCxNQUFNLE9BQU8seUJBQXlCOzt1SEFBekIseUJBQXlCOzJHQUF6Qix5QkFBeUIsZ0dBVDFCOzs7Ozs7O0dBT1Q7NEZBRVUseUJBQXlCO2tCQVhyQyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxzQkFBc0I7b0JBQ2hDLFFBQVEsRUFBRTs7Ozs7OztHQU9UO2lCQUNGOzhCQUVVLFNBQVM7c0JBQWpCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IENvbXBvbmVudFJlbmRlcmluZyB9IGZyb20gJ0BzaXRlY29yZS1qc3Mvc2l0ZWNvcmUtanNzL2xheW91dCc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ3NjLW1pc3NpbmctY29tcG9uZW50JyxcclxuICB0ZW1wbGF0ZTogYFxyXG4gICAgPGRpdlxyXG4gICAgICBzdHlsZT1cImJhY2tncm91bmQ6IGRhcmtvcmFuZ2U7IG91dGxpbmU6IDVweCBzb2xpZCBvcmFuZ2U7IHBhZGRpbmc6IDEwcHg7IGNvbG9yOiB3aGl0ZTsgbWF4LXdpZHRoOiA1MDBweDtcIlxyXG4gICAgPlxyXG4gICAgICA8aDI+e3sgcmVuZGVyaW5nLmNvbXBvbmVudE5hbWUgfX08L2gyPlxyXG4gICAgICA8cD5KU1MgY29tcG9uZW50IGlzIG1pc3NpbmcgQW5ndWxhciBjb21wb25lbnQgaW1wbGVtZW50YXRpb24uPC9wPlxyXG4gICAgPC9kaXY+XHJcbiAgYCxcclxufSlcclxuZXhwb3J0IGNsYXNzIE1pc3NpbmdDb21wb25lbnRDb21wb25lbnQge1xyXG4gIEBJbnB1dCgpIHJlbmRlcmluZzogQ29tcG9uZW50UmVuZGVyaW5nO1xyXG59XHJcbiJdfQ==
|