@piserve-tech/form-submission 1.3.260 → 1.3.262
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/esm2022/form-fields/iframe-fields/iframe-fields.component.mjs +90 -3
- package/esm2022/form-submission/submit-form/submit-form.component.mjs +2 -1
- package/esm2022/services/whenClause.service.mjs +9 -1
- package/fesm2022/piserve-tech-form-submission.mjs +98 -2
- package/fesm2022/piserve-tech-form-submission.mjs.map +1 -1
- package/form-fields/iframe-fields/iframe-fields.component.d.ts +4 -1
- package/package.json +1 -1
- package/services/whenClause.service.d.ts +1 -0
|
@@ -11,6 +11,28 @@ export class IframeFieldsComponent {
|
|
|
11
11
|
this.cssContent = '';
|
|
12
12
|
this.scriptContent = '';
|
|
13
13
|
this.urlContent = '';
|
|
14
|
+
this.embeddableDomains = [
|
|
15
|
+
// Generic sites that usually allow embedding
|
|
16
|
+
'youtube.com',
|
|
17
|
+
'youtu.be',
|
|
18
|
+
'vimeo.com',
|
|
19
|
+
'player.vimeo.com',
|
|
20
|
+
'maps.google.com',
|
|
21
|
+
'www.google.com/maps',
|
|
22
|
+
'openstreetmap.org',
|
|
23
|
+
'codesandbox.io',
|
|
24
|
+
'stackblitz.com',
|
|
25
|
+
'jsfiddle.net',
|
|
26
|
+
'codepen.io',
|
|
27
|
+
'spotify.com',
|
|
28
|
+
'soundcloud.com',
|
|
29
|
+
'figma.com',
|
|
30
|
+
'canva.com',
|
|
31
|
+
'notion.so',
|
|
32
|
+
'facebook.com/plugins',
|
|
33
|
+
'instagram.com/p',
|
|
34
|
+
'twitter.com',
|
|
35
|
+
];
|
|
14
36
|
}
|
|
15
37
|
ngOnInit() {
|
|
16
38
|
this.valueAssigned();
|
|
@@ -75,14 +97,79 @@ export class IframeFieldsComponent {
|
|
|
75
97
|
iframeDoc.close();
|
|
76
98
|
}
|
|
77
99
|
}
|
|
78
|
-
/** Render external links safely */
|
|
79
100
|
renderLink() {
|
|
80
|
-
|
|
101
|
+
if (!this.urlContent) {
|
|
102
|
+
this.safeUrl = '';
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
const url = this.urlContent.trim();
|
|
106
|
+
// Check if URL is valid
|
|
107
|
+
if (!this.isValidUrl(url)) {
|
|
108
|
+
console.warn('Invalid URL format');
|
|
109
|
+
this.safeUrl = '';
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
// Check if embeddable host
|
|
113
|
+
if (!this.isEmbeddableUrl(url)) {
|
|
114
|
+
console.warn('This website may not allow embedding');
|
|
115
|
+
// You can still LET it try
|
|
116
|
+
// Or block it completely
|
|
117
|
+
// For now: allow but warn
|
|
118
|
+
}
|
|
119
|
+
const embedUrl = this.convertToEmbedUrl(url);
|
|
120
|
+
this.safeUrl = this.sanitizeUrl(embedUrl);
|
|
121
|
+
}
|
|
122
|
+
isValidUrl(url) {
|
|
123
|
+
try {
|
|
124
|
+
new URL(url);
|
|
125
|
+
return true;
|
|
126
|
+
}
|
|
127
|
+
catch {
|
|
128
|
+
return false;
|
|
129
|
+
}
|
|
81
130
|
}
|
|
82
131
|
/** Sanitize URLs for iframes */
|
|
83
132
|
sanitizeUrl(url) {
|
|
84
133
|
return this.sanitizer.bypassSecurityTrustResourceUrl(url);
|
|
85
134
|
}
|
|
135
|
+
isEmbeddableUrl(url) {
|
|
136
|
+
try {
|
|
137
|
+
const parsed = new URL(url);
|
|
138
|
+
return this.embeddableDomains.some((domain) => parsed.hostname.includes(domain));
|
|
139
|
+
}
|
|
140
|
+
catch {
|
|
141
|
+
return false;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
convertToEmbedUrl(url) {
|
|
145
|
+
try {
|
|
146
|
+
const parsed = new URL(url);
|
|
147
|
+
// YouTube
|
|
148
|
+
if (parsed.hostname.includes('youtube.com') &&
|
|
149
|
+
parsed.searchParams.get('v')) {
|
|
150
|
+
return `https://www.youtube.com/embed/${parsed.searchParams.get('v')}`;
|
|
151
|
+
}
|
|
152
|
+
if (parsed.hostname.includes('youtu.be')) {
|
|
153
|
+
return `https://www.youtube.com/embed/${parsed.pathname.substring(1)}`;
|
|
154
|
+
}
|
|
155
|
+
// Vimeo
|
|
156
|
+
if (parsed.hostname.includes('vimeo.com')) {
|
|
157
|
+
const videoId = parsed.pathname.split('/')[1];
|
|
158
|
+
return `https://player.vimeo.com/video/${videoId}`;
|
|
159
|
+
}
|
|
160
|
+
// Google Maps (no API key needed)
|
|
161
|
+
if (parsed.hostname.includes('google.com') &&
|
|
162
|
+
parsed.pathname.startsWith('/maps')) {
|
|
163
|
+
if (parsed.pathname.includes('/maps/embed'))
|
|
164
|
+
return url;
|
|
165
|
+
return `https://www.google.com/maps/embed?${parsed.searchParams.toString()}`;
|
|
166
|
+
}
|
|
167
|
+
return url;
|
|
168
|
+
}
|
|
169
|
+
catch (e) {
|
|
170
|
+
return url;
|
|
171
|
+
}
|
|
172
|
+
}
|
|
86
173
|
/** Basic script validation to prevent unsafe operations */
|
|
87
174
|
isSafeScript(code) {
|
|
88
175
|
const unsafePatterns = [
|
|
@@ -108,4 +195,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
108
195
|
type: ViewChild,
|
|
109
196
|
args: ['iframeRef']
|
|
110
197
|
}] } });
|
|
111
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"iframe-fields.component.js","sourceRoot":"","sources":["../../../../../projects/form-submission/src/form-fields/iframe-fields/iframe-fields.component.ts","../../../../../projects/form-submission/src/form-fields/iframe-fields/iframe-fields.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, ViewChild } from '@angular/core';\nimport { DomSanitizer, SafeHtml, SafeResourceUrl } from '@angular/platform-browser';\n\n@Component({\n  selector: 'lib-iframe-fields',\n  templateUrl: './iframe-fields.component.html',\n  styleUrls: ['./iframe-fields.component.scss']\n})\nexport class IframeFieldsComponent {\n\n  @Input() question: any = {};\n  \n  height!: string;\n  width!: string;\n  contentType: string = '';\n  htmlContent: string = '';\n  cssContent: string = '';\n  scriptContent: string = '';\n  urlContent: string = '';\n  count!: number;\n\n  safeHtmlContent!: SafeHtml;\n  safeUrl!: SafeResourceUrl;\n\n  constructor(private sanitizer: DomSanitizer) {}\n\n  ngOnInit() {\n    this.valueAssigned();\n  }\n\n  ngAfterViewInit() {\n    if (this.contentType === 'HTML') {\n      setTimeout(() => this.renderHtml(), 0);\n    } else if (this.contentType === 'LINK') {\n      setTimeout(() => this.renderLink(), 0);\n    }\n  }\n\n  valueAssigned() {\n    this.count = this.question?.count;\n    this.width = this.question.formElement.appearance.width\n      ? this.question.formElement.appearance.width + 'px'\n      : '100%';\n    this.height = this.question.formElement.appearance.height\n      ? this.question.formElement.appearance.height + 'px'\n      : '100%';\n    this.contentType =\n      this.question.formElement.iFrameProperties.contentType;\n    this.htmlContent =\n      this.question.formElement.iFrameProperties.html.content;\n    this.cssContent =\n      this.question.formElement.iFrameProperties.style.css;\n    this.scriptContent =\n      this.question.formElement.iFrameProperties.script.code;\n    this.urlContent = this.question.formElement.iFrameProperties.url;\n  }\n  @ViewChild('iframeRef') iframeRef!: ElementRef<HTMLIFrameElement>;\n  /** Render HTML with dynamic CSS */\n  renderHtml() {\n    const iframe = this.iframeRef?.nativeElement;\n\n    if (!iframe) return;\n\n    const iframeDoc = iframe.contentDocument || iframe.contentWindow?.document;\n\n    if (iframeDoc) {\n      // Build the complete HTML content to inject into the iframe\n      const safeHtml = `\n        <html>\n          <head>\n            <style>${this.cssContent || ''}</style>\n          </head>\n          <body>\n            ${this.htmlContent || ''}\n            <script>\n              (function(localStorage) {\n                try {\n                  ${\n                    this.isSafeScript(this.scriptContent)\n                      ? this.scriptContent\n                      : ''\n                  }\n                } catch (e) {\n                  document.body.innerHTML += '<p style=\"color:red\">Script Error: ' + e.message + '</p>';\n                }\n              })(window.localStorage);\n            </script>\n          </body>\n        </html>\n      `;\n\n      iframeDoc.open();\n      iframeDoc.write(safeHtml);\n      iframeDoc.close();\n    }\n  }\n\n  /** Render external links safely */\n  renderLink() {\n    this.safeUrl = this.sanitizeUrl(this.urlContent);\n  }\n\n  /** Sanitize URLs for iframes */\n  sanitizeUrl(url: string): SafeResourceUrl {\n    return this.sanitizer.bypassSecurityTrustResourceUrl(url);\n  }\n\n  /** Basic script validation to prevent unsafe operations */\n  isSafeScript(code: string): boolean {\n    const unsafePatterns = [\n      /window\\.parent/,\n      /document\\.cookie/,\n      /fetch\\(/,\n      /XMLHttpRequest/,\n      /eval\\(/,\n      /new Function/,\n      /setTimeout|setInterval/,\n    ];\n\n    return !unsafePatterns.some((pattern) => pattern.test(code));\n  }\n}\n","<div class=\"custom-embed mb-4\"\n     [style.width]=\"width\"\n     [style.height]=\"height\">\n\n  <ng-container [ngSwitch]=\"contentType\">\n\n    <!-- HTML, CSS, and Script rendered together -->\n    <iframe\n      *ngSwitchCase=\"'HTML'\"\n      #iframeRef\n      sandbox=\"allow-scripts allow-same-origin\"\n      [style.width]=\"width\"\n      [style.height]=\"height\"\n      frameborder=\"0\"\n    ></iframe>\n\n    <!-- Link Embed -->\n     <div *ngSwitchCase=\"'LINK'\">\n     <iframe\n     *ngIf=\"safeUrl\"\n     [src]=\"safeUrl\"\n     [style.width]=\"width\"\n     [style.height]=\"height\"\n     frameborder=\"0\"\n     allowfullscreen>\n   </iframe>\n     </div>\n    \n\n  </ng-container>\n</div>\n"]}
|
|
198
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"iframe-fields.component.js","sourceRoot":"","sources":["../../../../../projects/form-submission/src/form-fields/iframe-fields/iframe-fields.component.ts","../../../../../projects/form-submission/src/form-fields/iframe-fields/iframe-fields.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, ViewChild } from '@angular/core';\nimport { DomSanitizer, SafeHtml, SafeResourceUrl } from '@angular/platform-browser';\n\n@Component({\n  selector: 'lib-iframe-fields',\n  templateUrl: './iframe-fields.component.html',\n  styleUrls: ['./iframe-fields.component.scss']\n})\nexport class IframeFieldsComponent {\n\n  @Input() question: any = {};\n  \n  height!: string;\n  width!: string;\n  contentType: string = '';\n  htmlContent: string = '';\n  cssContent: string = '';\n  scriptContent: string = '';\n  urlContent: string = '';\n  count!: number;\n\n  safeHtmlContent!: SafeHtml;\n  safeUrl!: SafeResourceUrl;\n\n  embeddableDomains = [\n    // Generic sites that usually allow embedding\n    'youtube.com',\n    'youtu.be',\n    'vimeo.com',\n    'player.vimeo.com',\n    'maps.google.com',\n    'www.google.com/maps',\n    'openstreetmap.org',\n    'codesandbox.io',\n    'stackblitz.com',\n    'jsfiddle.net',\n    'codepen.io',\n    'spotify.com',\n    'soundcloud.com',\n    'figma.com',\n    'canva.com',\n    'notion.so',\n    'facebook.com/plugins',\n    'instagram.com/p',\n    'twitter.com',\n  ];\n\n  constructor(private sanitizer: DomSanitizer) {}\n\n  ngOnInit() {\n    this.valueAssigned();\n  }\n\n  ngAfterViewInit() {\n    if (this.contentType === 'HTML') {\n      setTimeout(() => this.renderHtml(), 0);\n    } else if (this.contentType === 'LINK') {\n      setTimeout(() => this.renderLink(), 0);\n    }\n  }\n\n  valueAssigned() {\n    this.count = this.question?.count;\n    this.width = this.question.formElement.appearance.width\n      ? this.question.formElement.appearance.width + 'px'\n      : '100%';\n    this.height = this.question.formElement.appearance.height\n      ? this.question.formElement.appearance.height + 'px'\n      : '100%';\n    this.contentType =\n      this.question.formElement.iFrameProperties.contentType;\n    this.htmlContent =\n      this.question.formElement.iFrameProperties.html.content;\n    this.cssContent =\n      this.question.formElement.iFrameProperties.style.css;\n    this.scriptContent =\n      this.question.formElement.iFrameProperties.script.code;\n    this.urlContent = this.question.formElement.iFrameProperties.url;\n  }\n  @ViewChild('iframeRef') iframeRef!: ElementRef<HTMLIFrameElement>;\n  /** Render HTML with dynamic CSS */\n  renderHtml() {\n    const iframe = this.iframeRef?.nativeElement;\n\n    if (!iframe) return;\n\n    const iframeDoc = iframe.contentDocument || iframe.contentWindow?.document;\n\n    if (iframeDoc) {\n      // Build the complete HTML content to inject into the iframe\n      const safeHtml = `\n        <html>\n          <head>\n            <style>${this.cssContent || ''}</style>\n          </head>\n          <body>\n            ${this.htmlContent || ''}\n            <script>\n              (function(localStorage) {\n                try {\n                  ${\n                    this.isSafeScript(this.scriptContent)\n                      ? this.scriptContent\n                      : ''\n                  }\n                } catch (e) {\n                  document.body.innerHTML += '<p style=\"color:red\">Script Error: ' + e.message + '</p>';\n                }\n              })(window.localStorage);\n            </script>\n          </body>\n        </html>\n      `;\n\n      iframeDoc.open();\n      iframeDoc.write(safeHtml);\n      iframeDoc.close();\n    }\n  }\n\n  renderLink() {\n    if (!this.urlContent) {\n      this.safeUrl = '';\n      return;\n    }\n\n    const url = this.urlContent.trim();\n\n    // Check if URL is valid\n    if (!this.isValidUrl(url)) {\n      console.warn('Invalid URL format');\n      this.safeUrl = '';\n      return;\n    }\n\n    // Check if embeddable host\n    if (!this.isEmbeddableUrl(url)) {\n      console.warn('This website may not allow embedding');\n      // You can still LET it try\n      // Or block it completely\n      // For now: allow but warn\n    }\n\n    const embedUrl = this.convertToEmbedUrl(url);\n    this.safeUrl = this.sanitizeUrl(embedUrl);\n  }\n\n  isValidUrl(url: string): boolean {\n    try {\n      new URL(url);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /** Sanitize URLs for iframes */\n  sanitizeUrl(url: string): SafeResourceUrl {\n    return this.sanitizer.bypassSecurityTrustResourceUrl(url);\n  }\n\n  isEmbeddableUrl(url: string): boolean {\n    try {\n      const parsed = new URL(url);\n\n      return this.embeddableDomains.some((domain) =>\n        parsed.hostname.includes(domain)\n      );\n    } catch {\n      return false;\n    }\n  }\n\n  convertToEmbedUrl(url: string): string {\n    try {\n      const parsed = new URL(url);\n\n      // YouTube\n      if (\n        parsed.hostname.includes('youtube.com') &&\n        parsed.searchParams.get('v')\n      ) {\n        return `https://www.youtube.com/embed/${parsed.searchParams.get('v')}`;\n      }\n      if (parsed.hostname.includes('youtu.be')) {\n        return `https://www.youtube.com/embed/${parsed.pathname.substring(1)}`;\n      }\n\n      // Vimeo\n      if (parsed.hostname.includes('vimeo.com')) {\n        const videoId = parsed.pathname.split('/')[1];\n        return `https://player.vimeo.com/video/${videoId}`;\n      }\n\n      // Google Maps (no API key needed)\n      if (\n        parsed.hostname.includes('google.com') &&\n        parsed.pathname.startsWith('/maps')\n      ) {\n        if (parsed.pathname.includes('/maps/embed')) return url;\n\n        return `https://www.google.com/maps/embed?${parsed.searchParams.toString()}`;\n      }\n\n      return url;\n    } catch (e) {\n      return url;\n    }\n  }\n\n  /** Basic script validation to prevent unsafe operations */\n  isSafeScript(code: string): boolean {\n    const unsafePatterns = [\n      /window\\.parent/,\n      /document\\.cookie/,\n      /fetch\\(/,\n      /XMLHttpRequest/,\n      /eval\\(/,\n      /new Function/,\n      /setTimeout|setInterval/,\n    ];\n\n    return !unsafePatterns.some((pattern) => pattern.test(code));\n  }\n}\n","<div class=\"custom-embed mb-4\"\n     [style.width]=\"width\"\n     [style.height]=\"height\">\n\n  <ng-container [ngSwitch]=\"contentType\">\n\n    <!-- HTML, CSS, and Script rendered together -->\n    <iframe\n      *ngSwitchCase=\"'HTML'\"\n      #iframeRef\n      sandbox=\"allow-scripts allow-same-origin\"\n      [style.width]=\"width\"\n      [style.height]=\"height\"\n      frameborder=\"0\"\n    ></iframe>\n\n    <!-- Link Embed -->\n     <div *ngSwitchCase=\"'LINK'\">\n     <iframe\n     *ngIf=\"safeUrl\"\n     [src]=\"safeUrl\"\n     [style.width]=\"width\"\n     [style.height]=\"height\"\n     frameborder=\"0\"\n     allowfullscreen>\n   </iframe>\n     </div>\n    \n\n  </ng-container>\n</div>\n"]}
|