@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.
@@ -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
- this.safeUrl = this.sanitizeUrl(this.urlContent);
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":"AAAA,OAAO,EAAE,SAAS,EAAc,KAAK,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;;;;AAQxE,MAAM,OAAO,qBAAqB;IAgBhC,YAAoB,SAAuB;QAAvB,cAAS,GAAT,SAAS,CAAc;QAdlC,aAAQ,GAAQ,EAAE,CAAC;QAI5B,gBAAW,GAAW,EAAE,CAAC;QACzB,gBAAW,GAAW,EAAE,CAAC;QACzB,eAAU,GAAW,EAAE,CAAC;QACxB,kBAAa,GAAW,EAAE,CAAC;QAC3B,eAAU,GAAW,EAAE,CAAC;IAMsB,CAAC;IAE/C,QAAQ;QACN,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,WAAW,KAAK,MAAM,EAAE;YAC/B,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC,CAAC;SACxC;aAAM,IAAI,IAAI,CAAC,WAAW,KAAK,MAAM,EAAE;YACtC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC,CAAC;SACxC;IACH,CAAC;IAED,aAAa;QACX,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC;QAClC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK;YACrD,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,GAAG,IAAI;YACnD,CAAC,CAAC,MAAM,CAAC;QACX,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,UAAU,CAAC,MAAM;YACvD,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,UAAU,CAAC,MAAM,GAAG,IAAI;YACpD,CAAC,CAAC,MAAM,CAAC;QACX,IAAI,CAAC,WAAW;YACd,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,gBAAgB,CAAC,WAAW,CAAC;QACzD,IAAI,CAAC,WAAW;YACd,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC;QAC1D,IAAI,CAAC,UAAU;YACb,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,gBAAgB,CAAC,KAAK,CAAC,GAAG,CAAC;QACvD,IAAI,CAAC,aAAa;YAChB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC;QACzD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,gBAAgB,CAAC,GAAG,CAAC;IACnE,CAAC;IAED,mCAAmC;IACnC,UAAU;QACR,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,EAAE,aAAa,CAAC;QAE7C,IAAI,CAAC,MAAM;YAAE,OAAO;QAEpB,MAAM,SAAS,GAAG,MAAM,CAAC,eAAe,IAAI,MAAM,CAAC,aAAa,EAAE,QAAQ,CAAC;QAE3E,IAAI,SAAS,EAAE;YACb,4DAA4D;YAC5D,MAAM,QAAQ,GAAG;;;qBAGF,IAAI,CAAC,UAAU,IAAI,EAAE;;;cAG5B,IAAI,CAAC,WAAW,IAAI,EAAE;;;;oBAKhB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC;gBACnC,CAAC,CAAC,IAAI,CAAC,aAAa;gBACpB,CAAC,CAAC,EACN;;;;;;;;OAQX,CAAC;YAEF,SAAS,CAAC,IAAI,EAAE,CAAC;YACjB,SAAS,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAC1B,SAAS,CAAC,KAAK,EAAE,CAAC;SACnB;IACH,CAAC;IAED,mCAAmC;IACnC,UAAU;QACR,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACnD,CAAC;IAED,gCAAgC;IAChC,WAAW,CAAC,GAAW;QACrB,OAAO,IAAI,CAAC,SAAS,CAAC,8BAA8B,CAAC,GAAG,CAAC,CAAC;IAC5D,CAAC;IAED,2DAA2D;IAC3D,YAAY,CAAC,IAAY;QACvB,MAAM,cAAc,GAAG;YACrB,gBAAgB;YAChB,kBAAkB;YAClB,SAAS;YACT,gBAAgB;YAChB,QAAQ;YACR,cAAc;YACd,wBAAwB;SACzB,CAAC;QAEF,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IAC/D,CAAC;+GAhHU,qBAAqB;mGAArB,qBAAqB,mMCRlC,8sBA+BA;;4FDvBa,qBAAqB;kBALjC,SAAS;+BACE,mBAAmB;mGAMpB,QAAQ;sBAAhB,KAAK;gBA8CkB,SAAS;sBAAhC,SAAS;uBAAC,WAAW","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"]}