@colijnit/sharedcomponents 259.1.7 → 259.1.8

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.
@@ -18,22 +18,17 @@ export class CustomPdfDialogComponent {
18
18
  this.additionalFileChangeEvent = new EventEmitter();
19
19
  this.signaturePads = {};
20
20
  this.signatureCanvases = {};
21
+ this.fileBody = '';
22
+ this.fileStyle = '';
21
23
  this.showLoader = false;
24
+ // set to true if you want to skip email for debug purposes
25
+ this.enableLocalPreview = false;
22
26
  }
23
27
  showClass() {
24
28
  return true;
25
29
  }
26
30
  ngOnInit() {
27
- if (this.additionalFileContents) {
28
- const head = document.getElementsByTagName('head')[0];
29
- const styles = this.extractStyleParts(this.additionalFileContents);
30
- this.fileStyle = styles[0];
31
- const style = document.createElement('style');
32
- style.appendChild(document.createTextNode(this.fileStyle));
33
- head.appendChild(style);
34
- const body = this.extractBodyContents(this.additionalFileContents);
35
- this.fileBody = this._sanitizer.bypassSecurityTrustHtml(body[0]);
36
- }
31
+ this._processAdditionalFileContents();
37
32
  }
38
33
  ngAfterViewInit() {
39
34
  ['signatureCanvas', 'signatureCanvas2'].forEach(id => {
@@ -75,75 +70,144 @@ export class CustomPdfDialogComponent {
75
70
  }
76
71
  });
77
72
  }
73
+ _processAdditionalFileContents() {
74
+ var _a;
75
+ if (!this.additionalFileContents)
76
+ return;
77
+ const styles = this.extractStyleParts(this.additionalFileContents);
78
+ this.fileStyle = styles.join('\n').trim();
79
+ if (this.fileStyle) {
80
+ const head = document.getElementsByTagName('head')[0];
81
+ const styleEl = document.createElement('style');
82
+ styleEl.appendChild(document.createTextNode(this.fileStyle));
83
+ head.appendChild(styleEl);
84
+ }
85
+ const bodyParts = this.extractBodyContents(this.additionalFileContents);
86
+ const bodyHtml = ((_a = bodyParts[0]) !== null && _a !== void 0 ? _a : this.removeStyleAndScriptTags(this.additionalFileContents)).trim();
87
+ this.fileBody = this._sanitizer.bypassSecurityTrustHtml(bodyHtml);
88
+ }
78
89
  extractStyleParts(html) {
90
+ var _a;
79
91
  const stylePattern = /<style[^>]*>([\s\S]*?)<\/style>/gi;
80
92
  const matches = [];
81
93
  let match;
82
94
  while ((match = stylePattern.exec(html)) !== null) {
83
- matches.push(match[1].trim());
95
+ matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
84
96
  }
85
97
  return matches;
86
98
  }
87
99
  removeStyleAndScriptTags(html) {
88
100
  const styleTagPattern = /<style[^>]*>[\s\S]*?<\/style>/gi;
89
101
  const scriptTagPattern = /<script[^>]*>[\s\S]*?<\/script>/gi;
90
- return html.replace(styleTagPattern, '').replace(scriptTagPattern, '');
102
+ return (html || '').replace(styleTagPattern, '').replace(scriptTagPattern, '');
91
103
  }
92
104
  extractBodyContents(html) {
105
+ var _a;
93
106
  html = this.removeStyleAndScriptTags(html);
94
- const stylePattern = /<body[^>]*>([\s\S]*?)<\/body>/gi;
107
+ const bodyPattern = /<body[^>]*>([\s\S]*?)<\/body>/gi;
95
108
  const matches = [];
96
109
  let match;
97
- while ((match = stylePattern.exec(html)) !== null) {
98
- matches.push(match[1].trim());
110
+ while ((match = bodyPattern.exec(html)) !== null) {
111
+ matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
99
112
  }
100
113
  return matches;
101
114
  }
115
+ _isLocalhost() {
116
+ return ['localhost', '127.0.0.1', '::1'].includes(location.hostname);
117
+ }
102
118
  handleSaveClicked() {
119
+ var _a, _b;
103
120
  return __awaiter(this, void 0, void 0, function* () {
104
121
  this.showLoader = true;
105
122
  const clearButton = document.getElementById('clearButton');
106
- clearButton.hidden = true;
107
- const doc = new jsPDF();
108
- const htmlDocument = document.querySelector("#pdfBody");
109
- const documentSection = htmlDocument.getElementsByTagName('section');
110
- const sectionsCount = documentSection.length;
111
- for (let i = 0; i < sectionsCount; i++) {
112
- const html = documentSection[i];
123
+ if (clearButton)
124
+ clearButton.hidden = true;
125
+ const doc = new jsPDF({ unit: 'mm', format: 'a4' });
126
+ let createdTemp = false;
127
+ let root = ((_a = this.pdfBody) === null || _a === void 0 ? void 0 : _a.nativeElement) ||
128
+ document.querySelector('#pdfBody');
129
+ if (!root) {
130
+ root = document.createElement('div');
131
+ root.id = 'pdfBody';
132
+ root.style.position = 'fixed';
133
+ root.style.left = '-99999px';
134
+ root.style.top = '0';
135
+ root.style.width = '794px';
136
+ root.style.pointerEvents = 'none';
137
+ document.body.appendChild(root);
138
+ createdTemp = true;
139
+ }
140
+ if (this.additionalFileContents && !root.hasChildNodes()) {
141
+ const html = String(this.additionalFileContents);
142
+ if (html.includes('<html')) {
143
+ const parsed = new DOMParser().parseFromString(html, 'text/html');
144
+ root.innerHTML = ((_b = parsed.body) === null || _b === void 0 ? void 0 : _b.innerHTML) || html;
145
+ }
146
+ else {
147
+ root.innerHTML = html;
148
+ }
149
+ }
150
+ const allSections = Array.from(root.querySelectorAll('section'));
151
+ const leafSections = allSections.filter(s => !s.querySelector('section'));
152
+ const nodesToRender = leafSections.length ? leafSections : (allSections.length ? allSections : [root]);
153
+ const pageWidth = doc.internal.pageSize.getWidth();
154
+ const pageHeight = doc.internal.pageSize.getHeight();
155
+ for (let i = 0; i < nodesToRender.length; i++) {
156
+ const el = nodesToRender[i];
113
157
  try {
114
- const dataUrl = yield toJpeg(html, {
115
- quality: 1,
116
- backgroundColor: 'white'
117
- });
118
- const img = new Image();
119
- img.src = dataUrl;
158
+ const dataUrl = yield toJpeg(el, { quality: 1, backgroundColor: 'white' });
120
159
  yield new Promise((resolve) => {
160
+ const img = new Image();
121
161
  img.onload = () => {
122
- const imgWidth = 218;
123
- const imgHeight = (img.height * imgWidth) / img.width;
124
- doc.addImage(dataUrl, 'JPG', -4, 0, imgWidth, imgHeight);
125
- if (i < sectionsCount - 1) {
162
+ const w = img.width;
163
+ const h = img.height;
164
+ const scale = Math.min(pageWidth / w, pageHeight / h);
165
+ const renderW = w * scale;
166
+ const renderH = h * scale;
167
+ doc.addImage(dataUrl, 'JPG', 0, 0, renderW, renderH);
168
+ if (i < nodesToRender.length - 1)
126
169
  doc.addPage();
127
- }
128
170
  resolve();
129
171
  };
172
+ img.src = dataUrl;
130
173
  });
131
174
  }
132
- catch (error) {
133
- console.error('Error generating image:', error);
175
+ catch (err) {
176
+ console.error('Error generating image:', err);
134
177
  }
135
178
  }
136
- const tempFile = doc.output('datauristring');
137
- const file = this.dataURItoBlob(tempFile);
138
- const fileAsCoDocument = yield FileUtils.ReadFileAsNewCoDocument(file);
139
- fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType.Image : FileType.Document;
140
- fileAsCoDocument.creationDate = new Date();
141
- fileAsCoDocument.modifiedDate = new Date();
142
- fileAsCoDocument.reports = 1;
143
- this.additionalFileChangeEvent.emit(fileAsCoDocument);
144
- this.showLoader = false;
145
- this.closePDFDialog.emit();
146
- clearButton.hidden = false;
179
+ if (createdTemp && root && root.parentNode) {
180
+ root.parentNode.removeChild(root);
181
+ }
182
+ // enableLocalPreview can be enabled to see the output without sending email
183
+ if (this.enableLocalPreview && this._isLocalhost()) {
184
+ const blob = doc.output('blob');
185
+ const url = URL.createObjectURL(blob);
186
+ window.open(url, '_blank');
187
+ this.showLoader = false;
188
+ if (clearButton)
189
+ clearButton.hidden = false;
190
+ return;
191
+ }
192
+ try {
193
+ const tempFile = doc.output('datauristring');
194
+ const file = this.dataURItoBlob(tempFile);
195
+ const fileAsCoDocument = yield FileUtils.ReadFileAsNewCoDocument(file);
196
+ fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType.Image : FileType.Document;
197
+ fileAsCoDocument.creationDate = new Date();
198
+ fileAsCoDocument.modifiedDate = new Date();
199
+ fileAsCoDocument.reports = 1;
200
+ this.additionalFileChangeEvent.emit(fileAsCoDocument);
201
+ }
202
+ catch (e) {
203
+ console.error('PDF packaging failed:', e);
204
+ }
205
+ finally {
206
+ this.showLoader = false;
207
+ this.closePDFDialog.emit();
208
+ if (clearButton)
209
+ clearButton.hidden = false;
210
+ }
147
211
  });
148
212
  }
149
213
  dataURItoBlob(dataURI) {
@@ -181,7 +245,7 @@ CustomPdfDialogComponent.decorators = [
181
245
  <div *ngIf="showLoader" class="loader-container">
182
246
  <co-loader *ngIf="showLoader"></co-loader>
183
247
  </div>
184
-
248
+
185
249
  <div #pdfBody id="pdfBody">
186
250
  <section [innerHTML]="fileBody"></section>
187
251
  </div>
@@ -212,4 +276,4 @@ CustomPdfDialogComponent.propDecorators = {
212
276
  closePDFDialog: [{ type: Output }],
213
277
  additionalFileChangeEvent: [{ type: Output }]
214
278
  };
215
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-pdf-dialog.component.js","sourceRoot":"","sources":["../../../../../../projects/sharedcomponents/src/lib/components/custom-pdf/custom-pdf-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Component, ElementRef, EventEmitter, HostBinding, Input, OnInit, Output, ViewChild, ViewEncapsulation} from '@angular/core';\r\nimport {jsPDF} from 'jspdf';\r\nimport {IconCacheService} from '../../service/icon-cache.service';\r\nimport {DomSanitizer} from '@angular/platform-browser';\r\nimport SignaturePad from 'signature_pad';\r\nimport {toJpeg} from 'html-to-image';\r\nimport {CoDocument} from '@colijnit/mainapi/build/model/co-document.bo';\r\nimport {FileUtils} from '@colijnit/mainapi/build/utils/file-utils';\r\n\r\nimport {FileType} from '@colijnit/mainapi/build/enum/file-type.enum';\r\nimport {Icon} from '../../enum/icon.enum';\r\n\r\n@Component({\r\n  selector: 'co-custom-pdf-dialog',\r\n  template: `\r\n    <co-dialog\r\n      [footerTemplate]=\"footerTemplate\"\r\n      id=\"custom-pdf-dialog\"\r\n      (closeClick)=\"closePDFDialog.emit()\">\r\n      <!-- create a nice container for loader so it is not random in the corner somewhere -->\r\n      <div *ngIf=\"showLoader\" class=\"loader-container\">\r\n        <co-loader *ngIf=\"showLoader\"></co-loader>\r\n      </div>\r\n      \r\n      <div #pdfBody id=\"pdfBody\">\r\n        <section [innerHTML]=\"fileBody\"></section>\r\n      </div>\r\n      <ng-template #footerTemplate>\r\n        <div class=\"co-dialog-footer-button-wrapper\">\r\n          <co-button class=\"save-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CheckDuotone)\"\r\n                     (click)=\"handleSaveClicked()\"></co-button>\r\n          <co-button class=\"close-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CrossSkinny)\"\r\n                     (click)=\"closePDFDialog.emit()\"></co-button>\r\n        </div>\r\n      </ng-template>\r\n    </co-dialog>\r\n  `,\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class CustomPdfDialogComponent implements OnInit, AfterViewInit {\r\n  public readonly icons: typeof Icon = Icon;\r\n\r\n  @HostBinding('class.co-custom-pdf-dialog')\r\n  public showClass() {\r\n    return true;\r\n  }\r\n\r\n  @ViewChild('pdfBody')\r\n  public pdfBody: ElementRef;\r\n\r\n  @Input()\r\n  public additionalFileContents: any;\r\n\r\n  @Input()\r\n  public additionalFileName: string = '';\r\n\r\n  @Output()\r\n  public closePDFDialog: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n  @Output()\r\n  public additionalFileChangeEvent: EventEmitter<CoDocument> = new EventEmitter<CoDocument>();\r\n\r\n  public canvas: HTMLCanvasElement;\r\n  public context: CanvasRenderingContext2D;\r\n  public signaturePads: { [key: string]: SignaturePad } = {};\r\n  public signatureCanvases: { [key: string]: ElementRef } = {};\r\n  public fileBody: any;\r\n  public fileStyle: string;\r\n\r\n  public showLoader: boolean = false;\r\n\r\n  constructor(\r\n    public iconCacheService: IconCacheService,\r\n    private _sanitizer: DomSanitizer\r\n  ) {\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (this.additionalFileContents) {\r\n      const head: HTMLHeadElement = document.getElementsByTagName('head')[0];\r\n      const styles: string[] = this.extractStyleParts(this.additionalFileContents);\r\n      this.fileStyle = styles[0];\r\n      const style: HTMLStyleElement = document.createElement('style');\r\n      style.appendChild(document.createTextNode(this.fileStyle));\r\n      head.appendChild(style);\r\n      const body: string[] = this.extractBodyContents(this.additionalFileContents);\r\n      this.fileBody = this._sanitizer.bypassSecurityTrustHtml(body[0]);\r\n    }\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    ['signatureCanvas', 'signatureCanvas2'].forEach(id => {\r\n      const canvasElement = document.getElementById(id);\r\n      if (canvasElement) {\r\n        const canvasRef = new ElementRef(canvasElement);\r\n        const signaturePad = new SignaturePad(<HTMLCanvasElement>canvasRef.nativeElement);\r\n        this.signatureCanvases[id] = canvasRef;\r\n        this.signaturePads[id] = signaturePad;\r\n\r\n        setTimeout(() => {\r\n          this._resizeSignatureCanvas(signaturePad, canvasRef);\r\n        });\r\n      }\r\n    });\r\n    const clearButton = document.getElementById('clearButton');\r\n    clearButton?.addEventListener('click', () => this.handleClear());\r\n\r\n    // Due to a bug or known issue with html-to-image we need a bit of JS the fill in the checked value when selecting a checkbox.\r\n    // Handling the checkboxes and radiobuttons\r\n    const container: HTMLElement = this.pdfBody.nativeElement as HTMLElement;\r\n    container.addEventListener('change', (event: Event) => {\r\n      const target = event.target as HTMLElement;\r\n      if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'checkbox') {\r\n        const checkbox: HTMLInputElement = target as HTMLInputElement;\r\n        if (checkbox.checked) {\r\n          checkbox.setAttribute('checked', 'true');\r\n        } else {\r\n          checkbox.removeAttribute('checked');\r\n        }\r\n      } else if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'radio') {\r\n        const radio: HTMLInputElement = target as HTMLInputElement;\r\n        const groupName: string = radio.name;\r\n        const radios: NodeListOf<Element> = container.querySelectorAll(`input[type=\"radio\"][name=\"${groupName}\"]`);\r\n        radios.forEach((radio) => radio.removeAttribute('checked'));\r\n        if (radio.checked) {\r\n          radio.setAttribute('checked', 'true');\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  public extractStyleParts(html: string): string[] {\r\n    const stylePattern: RegExp = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = stylePattern.exec(html)) !== null) {\r\n      matches.push(match[1].trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  public removeStyleAndScriptTags(html: string): string {\r\n    const styleTagPattern: RegExp = /<style[^>]*>[\\s\\S]*?<\\/style>/gi;\r\n    const scriptTagPattern: RegExp = /<script[^>]*>[\\s\\S]*?<\\/script>/gi;\r\n    return html.replace(styleTagPattern, '').replace(scriptTagPattern, '');\r\n  }\r\n\r\n  public extractBodyContents(html: string): string [] {\r\n    html = this.removeStyleAndScriptTags(html);\r\n    const stylePattern: RegExp = /<body[^>]*>([\\s\\S]*?)<\\/body>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = stylePattern.exec(html)) !== null) {\r\n      matches.push(match[1].trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  public async handleSaveClicked(): Promise<void> {\r\n    this.showLoader = true;\r\n    const clearButton: HTMLElement = document.getElementById('clearButton');\r\n    clearButton.hidden = true;\r\n    const doc: jsPDF = new jsPDF();\r\n    const htmlDocument: HTMLElement = document.querySelector(\"#pdfBody\");\r\n    const documentSection: HTMLCollectionOf<HTMLElement> = htmlDocument.getElementsByTagName('section');\r\n    const sectionsCount: number = documentSection.length;\r\n\r\n    for (let i: number = 0; i < sectionsCount; i++) {\r\n      const html: HTMLElement = documentSection[i];\r\n      try {\r\n        const dataUrl: string = await toJpeg(html, {\r\n          quality: 1,\r\n          backgroundColor: 'white'\r\n        });\r\n        const img: HTMLImageElement = new Image();\r\n        img.src = dataUrl;\r\n        await new Promise<void>((resolve) => {\r\n          img.onload = () => {\r\n            const imgWidth: number = 218;\r\n            const imgHeight: number = (img.height * imgWidth) / img.width;\r\n            doc.addImage(dataUrl, 'JPG', -4, 0, imgWidth, imgHeight);\r\n            if (i < sectionsCount - 1) {\r\n              doc.addPage();\r\n            }\r\n            resolve();\r\n          };\r\n        });\r\n      } catch (error) {\r\n        console.error('Error generating image:', error);\r\n      }\r\n    }\r\n    const tempFile: string = doc.output('datauristring');\r\n    const file: File = this.dataURItoBlob(tempFile);\r\n    const fileAsCoDocument = await FileUtils.ReadFileAsNewCoDocument(file);\r\n    fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType.Image : FileType.Document;\r\n    fileAsCoDocument.creationDate = new Date();\r\n    fileAsCoDocument.modifiedDate = new Date();\r\n    fileAsCoDocument.reports = 1;\r\n    this.additionalFileChangeEvent.emit(fileAsCoDocument);\r\n    this.showLoader = false;\r\n    this.closePDFDialog.emit();\r\n    clearButton.hidden = false;\r\n  }\r\n\r\n  public dataURItoBlob(dataURI): File {\r\n    const byteString: string = atob(dataURI.split(',')[1]);\r\n    const ab: ArrayBuffer = new ArrayBuffer(byteString.length);\r\n    const ia: Uint8Array = new Uint8Array(ab);\r\n    for (let i: number = 0; i < byteString.length; i++) {\r\n      ia[i] = byteString.charCodeAt(i);\r\n    }\r\n    return new File([ab], this.additionalFileName + '.pdf');\r\n  }\r\n\r\n  public handleClear(): void {\r\n    Object.values(this.signaturePads).forEach(signaturePad => {\r\n      signaturePad?.clear();\r\n    });\r\n  }\r\n\r\n  private _resizeSignatureCanvas(signature: SignaturePad, signatureCanvas: ElementRef): void {\r\n    const imageBeforeResize: string = signature.toDataURL();\r\n    const ratio: number = Math.max(window.devicePixelRatio || 1, 1);\r\n    signatureCanvas.nativeElement.width = signatureCanvas.nativeElement.offsetWidth * ratio;\r\n    signatureCanvas.nativeElement.height = signatureCanvas.nativeElement.offsetHeight * ratio;\r\n    signatureCanvas.nativeElement.getContext(\"2d\").scale(ratio, ratio);\r\n    signature.fromDataURL(imageBeforeResize);\r\n  }\r\n}\r\n"]}
279
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-pdf-dialog.component.js","sourceRoot":"","sources":["../../../../../../projects/sharedcomponents/src/lib/components/custom-pdf/custom-pdf-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Component, ElementRef, EventEmitter, HostBinding, Input, OnInit, Output, ViewChild, ViewEncapsulation} from '@angular/core';\r\nimport {jsPDF} from 'jspdf';\r\nimport {IconCacheService} from '../../service/icon-cache.service';\r\nimport {DomSanitizer} from '@angular/platform-browser';\r\nimport SignaturePad from 'signature_pad';\r\nimport {toJpeg} from 'html-to-image';\r\nimport {CoDocument} from '@colijnit/mainapi/build/model/co-document.bo';\r\nimport {FileUtils} from '@colijnit/mainapi/build/utils/file-utils';\r\n\r\nimport {FileType} from '@colijnit/mainapi/build/enum/file-type.enum';\r\nimport {Icon} from '../../enum/icon.enum';\r\n\r\n@Component({\r\n  selector: 'co-custom-pdf-dialog',\r\n  template: `\r\n    <co-dialog\r\n      [footerTemplate]=\"footerTemplate\"\r\n      id=\"custom-pdf-dialog\"\r\n      (closeClick)=\"closePDFDialog.emit()\">\r\n      <!-- create a nice container for loader so it is not random in the corner somewhere -->\r\n      <div *ngIf=\"showLoader\" class=\"loader-container\">\r\n        <co-loader *ngIf=\"showLoader\"></co-loader>\r\n      </div>\r\n\r\n      <div #pdfBody id=\"pdfBody\">\r\n        <section [innerHTML]=\"fileBody\"></section>\r\n      </div>\r\n      <ng-template #footerTemplate>\r\n        <div class=\"co-dialog-footer-button-wrapper\">\r\n          <co-button class=\"save-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CheckDuotone)\"\r\n                     (click)=\"handleSaveClicked()\"></co-button>\r\n          <co-button class=\"close-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CrossSkinny)\"\r\n                     (click)=\"closePDFDialog.emit()\"></co-button>\r\n        </div>\r\n      </ng-template>\r\n    </co-dialog>\r\n  `,\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class CustomPdfDialogComponent implements OnInit, AfterViewInit {\r\n  public readonly icons: typeof Icon = Icon;\r\n\r\n  @HostBinding('class.co-custom-pdf-dialog')\r\n  public showClass() {\r\n    return true;\r\n  }\r\n\r\n  @ViewChild('pdfBody')\r\n  public pdfBody: ElementRef;\r\n\r\n  @Input()\r\n  public additionalFileContents: any;\r\n\r\n  @Input()\r\n  public additionalFileName: string = '';\r\n\r\n  @Output()\r\n  public closePDFDialog: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n  @Output()\r\n  public additionalFileChangeEvent: EventEmitter<CoDocument> = new EventEmitter<CoDocument>();\r\n\r\n  public canvas: HTMLCanvasElement;\r\n  public context: CanvasRenderingContext2D;\r\n  public signaturePads: { [key: string]: SignaturePad } = {};\r\n  public signatureCanvases: { [key: string]: ElementRef } = {};\r\n  public fileBody: any = '';\r\n  public fileStyle: string = '';\r\n\r\n  public showLoader: boolean = false;\r\n\r\n  // set to true if you want to skip email for debug purposes\r\n  private enableLocalPreview = false;\r\n\r\n  constructor(\r\n    public iconCacheService: IconCacheService,\r\n    private _sanitizer: DomSanitizer\r\n  ) {\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this._processAdditionalFileContents();\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    ['signatureCanvas', 'signatureCanvas2'].forEach(id => {\r\n      const canvasElement = document.getElementById(id);\r\n      if (canvasElement) {\r\n        const canvasRef = new ElementRef(canvasElement);\r\n        const signaturePad = new SignaturePad(<HTMLCanvasElement>canvasRef.nativeElement);\r\n        this.signatureCanvases[id] = canvasRef;\r\n        this.signaturePads[id] = signaturePad;\r\n\r\n        setTimeout(() => {\r\n          this._resizeSignatureCanvas(signaturePad, canvasRef);\r\n        });\r\n      }\r\n    });\r\n    const clearButton = document.getElementById('clearButton');\r\n    clearButton?.addEventListener('click', () => this.handleClear());\r\n\r\n    // Due to a bug or known issue with html-to-image we need a bit of JS the fill in the checked value when selecting a checkbox.\r\n    // Handling the checkboxes and radiobuttons\r\n    const container: HTMLElement = this.pdfBody.nativeElement as HTMLElement;\r\n    container.addEventListener('change', (event: Event) => {\r\n      const target = event.target as HTMLElement;\r\n      if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'checkbox') {\r\n        const checkbox: HTMLInputElement = target as HTMLInputElement;\r\n        if (checkbox.checked) {\r\n          checkbox.setAttribute('checked', 'true');\r\n        } else {\r\n          checkbox.removeAttribute('checked');\r\n        }\r\n      } else if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'radio') {\r\n        const radio: HTMLInputElement = target as HTMLInputElement;\r\n        const groupName: string = radio.name;\r\n        const radios: NodeListOf<Element> = container.querySelectorAll(`input[type=\"radio\"][name=\"${groupName}\"]`);\r\n        radios.forEach((radio) => radio.removeAttribute('checked'));\r\n        if (radio.checked) {\r\n          radio.setAttribute('checked', 'true');\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private _processAdditionalFileContents(): void {\r\n    if (!this.additionalFileContents) return;\r\n\r\n    const styles = this.extractStyleParts(this.additionalFileContents);\r\n    this.fileStyle = styles.join('\\n').trim();\r\n    if (this.fileStyle) {\r\n      const head = document.getElementsByTagName('head')[0];\r\n      const styleEl = document.createElement('style');\r\n      styleEl.appendChild(document.createTextNode(this.fileStyle));\r\n      head.appendChild(styleEl);\r\n    }\r\n\r\n    const bodyParts = this.extractBodyContents(this.additionalFileContents);\r\n    const bodyHtml = (bodyParts[0] ?? this.removeStyleAndScriptTags(this.additionalFileContents)).trim();\r\n    this.fileBody = this._sanitizer.bypassSecurityTrustHtml(bodyHtml);\r\n  }\r\n\r\n  public extractStyleParts(html: string): string[] {\r\n    const stylePattern = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = stylePattern.exec(html)) !== null) {\r\n      matches.push((match[1] ?? '').trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  public removeStyleAndScriptTags(html: string): string {\r\n    const styleTagPattern = /<style[^>]*>[\\s\\S]*?<\\/style>/gi;\r\n    const scriptTagPattern = /<script[^>]*>[\\s\\S]*?<\\/script>/gi;\r\n    return (html || '').replace(styleTagPattern, '').replace(scriptTagPattern, '');\r\n  }\r\n\r\n  public extractBodyContents(html: string): string[] {\r\n    html = this.removeStyleAndScriptTags(html);\r\n    const bodyPattern = /<body[^>]*>([\\s\\S]*?)<\\/body>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = bodyPattern.exec(html)) !== null) {\r\n      matches.push((match[1] ?? '').trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  private _isLocalhost(): boolean {\r\n    return ['localhost', '127.0.0.1', '::1'].includes(location.hostname);\r\n  }\r\n\r\n  public async handleSaveClicked(): Promise<void> {\r\n    this.showLoader = true;\r\n\r\n    const clearButton = document.getElementById('clearButton');\r\n    if (clearButton) clearButton.hidden = true;\r\n\r\n    const doc = new jsPDF({ unit: 'mm', format: 'a4' });\r\n\r\n    let createdTemp = false;\r\n    let root: HTMLElement | null =\r\n      (this.pdfBody?.nativeElement as HTMLElement) ||\r\n      (document.querySelector('#pdfBody') as HTMLElement);\r\n\r\n    if (!root) {\r\n      root = document.createElement('div');\r\n      root.id = 'pdfBody';\r\n      root.style.position = 'fixed';\r\n      root.style.left = '-99999px';\r\n      root.style.top = '0';\r\n      root.style.width = '794px';\r\n      root.style.pointerEvents = 'none';\r\n      document.body.appendChild(root);\r\n      createdTemp = true;\r\n    }\r\n\r\n    if (this.additionalFileContents && !root.hasChildNodes()) {\r\n      const html = String(this.additionalFileContents);\r\n      if (html.includes('<html')) {\r\n        const parsed = new DOMParser().parseFromString(html, 'text/html');\r\n        root.innerHTML = parsed.body?.innerHTML || html;\r\n      } else {\r\n        root.innerHTML = html;\r\n      }\r\n    }\r\n\r\n    const allSections = Array.from(root.querySelectorAll('section'));\r\n    const leafSections = allSections.filter(s => !s.querySelector('section'));\r\n    const nodesToRender: HTMLElement[] =\r\n      leafSections.length ? leafSections : (allSections.length ? allSections : [root]);\r\n\r\n    const pageWidth = doc.internal.pageSize.getWidth();\r\n    const pageHeight = doc.internal.pageSize.getHeight();\r\n\r\n    for (let i = 0; i < nodesToRender.length; i++) {\r\n      const el = nodesToRender[i];\r\n      try {\r\n        const dataUrl = await toJpeg(el, { quality: 1, backgroundColor: 'white' });\r\n        await new Promise<void>((resolve) => {\r\n          const img = new Image();\r\n          img.onload = () => {\r\n            const w = img.width;\r\n            const h = img.height;\r\n            const scale = Math.min(pageWidth / w, pageHeight / h);\r\n            const renderW = w * scale;\r\n            const renderH = h * scale;\r\n\r\n            doc.addImage(dataUrl, 'JPG', 0, 0, renderW, renderH);\r\n            if (i < nodesToRender.length - 1) doc.addPage();\r\n            resolve();\r\n          };\r\n          img.src = dataUrl;\r\n        });\r\n      } catch (err) {\r\n        console.error('Error generating image:', err);\r\n      }\r\n    }\r\n\r\n    if (createdTemp && root && root.parentNode) {\r\n      root.parentNode.removeChild(root);\r\n    }\r\n\r\n    // enableLocalPreview can be enabled to see the output without sending email\r\n    if (this.enableLocalPreview && this._isLocalhost()) {\r\n      const blob = doc.output('blob');\r\n      const url = URL.createObjectURL(blob);\r\n      window.open(url as unknown as string, '_blank');\r\n      this.showLoader = false;\r\n      if (clearButton) clearButton.hidden = false;\r\n      return;\r\n    }\r\n\r\n    try {\r\n      const tempFile = doc.output('datauristring');\r\n      const file = this.dataURItoBlob(tempFile);\r\n      const fileAsCoDocument = await FileUtils.ReadFileAsNewCoDocument(file);\r\n      fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType.Image : FileType.Document;\r\n      fileAsCoDocument.creationDate = new Date();\r\n      fileAsCoDocument.modifiedDate = new Date();\r\n      fileAsCoDocument.reports = 1;\r\n      this.additionalFileChangeEvent.emit(fileAsCoDocument);\r\n    } catch (e) {\r\n      console.error('PDF packaging failed:', e);\r\n    } finally {\r\n      this.showLoader = false;\r\n      this.closePDFDialog.emit();\r\n      if (clearButton) clearButton.hidden = false;\r\n    }\r\n  }\r\n\r\n  public dataURItoBlob(dataURI): File {\r\n    const byteString: string = atob(dataURI.split(',')[1]);\r\n    const ab: ArrayBuffer = new ArrayBuffer(byteString.length);\r\n    const ia: Uint8Array = new Uint8Array(ab);\r\n    for (let i: number = 0; i < byteString.length; i++) {\r\n      ia[i] = byteString.charCodeAt(i);\r\n    }\r\n    return new File([ab], this.additionalFileName + '.pdf');\r\n  }\r\n\r\n  public handleClear(): void {\r\n    Object.values(this.signaturePads).forEach(signaturePad => {\r\n      signaturePad?.clear();\r\n    });\r\n  }\r\n\r\n  private _resizeSignatureCanvas(signature: SignaturePad, signatureCanvas: ElementRef): void {\r\n    const imageBeforeResize: string = signature.toDataURL();\r\n    const ratio: number = Math.max(window.devicePixelRatio || 1, 1);\r\n    signatureCanvas.nativeElement.width = signatureCanvas.nativeElement.offsetWidth * ratio;\r\n    signatureCanvas.nativeElement.height = signatureCanvas.nativeElement.offsetHeight * ratio;\r\n    signatureCanvas.nativeElement.getContext(\"2d\").scale(ratio, ratio);\r\n    signature.fromDataURL(imageBeforeResize);\r\n  }\r\n}\r\n"]}
package/favicon.ico ADDED
Binary file
@@ -3488,22 +3488,17 @@ class CustomPdfDialogComponent {
3488
3488
  this.additionalFileChangeEvent = new EventEmitter();
3489
3489
  this.signaturePads = {};
3490
3490
  this.signatureCanvases = {};
3491
+ this.fileBody = '';
3492
+ this.fileStyle = '';
3491
3493
  this.showLoader = false;
3494
+ // set to true if you want to skip email for debug purposes
3495
+ this.enableLocalPreview = false;
3492
3496
  }
3493
3497
  showClass() {
3494
3498
  return true;
3495
3499
  }
3496
3500
  ngOnInit() {
3497
- if (this.additionalFileContents) {
3498
- const head = document.getElementsByTagName('head')[0];
3499
- const styles = this.extractStyleParts(this.additionalFileContents);
3500
- this.fileStyle = styles[0];
3501
- const style = document.createElement('style');
3502
- style.appendChild(document.createTextNode(this.fileStyle));
3503
- head.appendChild(style);
3504
- const body = this.extractBodyContents(this.additionalFileContents);
3505
- this.fileBody = this._sanitizer.bypassSecurityTrustHtml(body[0]);
3506
- }
3501
+ this._processAdditionalFileContents();
3507
3502
  }
3508
3503
  ngAfterViewInit() {
3509
3504
  ['signatureCanvas', 'signatureCanvas2'].forEach(id => {
@@ -3545,75 +3540,144 @@ class CustomPdfDialogComponent {
3545
3540
  }
3546
3541
  });
3547
3542
  }
3543
+ _processAdditionalFileContents() {
3544
+ var _a;
3545
+ if (!this.additionalFileContents)
3546
+ return;
3547
+ const styles = this.extractStyleParts(this.additionalFileContents);
3548
+ this.fileStyle = styles.join('\n').trim();
3549
+ if (this.fileStyle) {
3550
+ const head = document.getElementsByTagName('head')[0];
3551
+ const styleEl = document.createElement('style');
3552
+ styleEl.appendChild(document.createTextNode(this.fileStyle));
3553
+ head.appendChild(styleEl);
3554
+ }
3555
+ const bodyParts = this.extractBodyContents(this.additionalFileContents);
3556
+ const bodyHtml = ((_a = bodyParts[0]) !== null && _a !== void 0 ? _a : this.removeStyleAndScriptTags(this.additionalFileContents)).trim();
3557
+ this.fileBody = this._sanitizer.bypassSecurityTrustHtml(bodyHtml);
3558
+ }
3548
3559
  extractStyleParts(html) {
3560
+ var _a;
3549
3561
  const stylePattern = /<style[^>]*>([\s\S]*?)<\/style>/gi;
3550
3562
  const matches = [];
3551
3563
  let match;
3552
3564
  while ((match = stylePattern.exec(html)) !== null) {
3553
- matches.push(match[1].trim());
3565
+ matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
3554
3566
  }
3555
3567
  return matches;
3556
3568
  }
3557
3569
  removeStyleAndScriptTags(html) {
3558
3570
  const styleTagPattern = /<style[^>]*>[\s\S]*?<\/style>/gi;
3559
3571
  const scriptTagPattern = /<script[^>]*>[\s\S]*?<\/script>/gi;
3560
- return html.replace(styleTagPattern, '').replace(scriptTagPattern, '');
3572
+ return (html || '').replace(styleTagPattern, '').replace(scriptTagPattern, '');
3561
3573
  }
3562
3574
  extractBodyContents(html) {
3575
+ var _a;
3563
3576
  html = this.removeStyleAndScriptTags(html);
3564
- const stylePattern = /<body[^>]*>([\s\S]*?)<\/body>/gi;
3577
+ const bodyPattern = /<body[^>]*>([\s\S]*?)<\/body>/gi;
3565
3578
  const matches = [];
3566
3579
  let match;
3567
- while ((match = stylePattern.exec(html)) !== null) {
3568
- matches.push(match[1].trim());
3580
+ while ((match = bodyPattern.exec(html)) !== null) {
3581
+ matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
3569
3582
  }
3570
3583
  return matches;
3571
3584
  }
3585
+ _isLocalhost() {
3586
+ return ['localhost', '127.0.0.1', '::1'].includes(location.hostname);
3587
+ }
3572
3588
  handleSaveClicked() {
3589
+ var _a, _b;
3573
3590
  return __awaiter(this, void 0, void 0, function* () {
3574
3591
  this.showLoader = true;
3575
3592
  const clearButton = document.getElementById('clearButton');
3576
- clearButton.hidden = true;
3577
- const doc = new jsPDF();
3578
- const htmlDocument = document.querySelector("#pdfBody");
3579
- const documentSection = htmlDocument.getElementsByTagName('section');
3580
- const sectionsCount = documentSection.length;
3581
- for (let i = 0; i < sectionsCount; i++) {
3582
- const html = documentSection[i];
3593
+ if (clearButton)
3594
+ clearButton.hidden = true;
3595
+ const doc = new jsPDF({ unit: 'mm', format: 'a4' });
3596
+ let createdTemp = false;
3597
+ let root = ((_a = this.pdfBody) === null || _a === void 0 ? void 0 : _a.nativeElement) ||
3598
+ document.querySelector('#pdfBody');
3599
+ if (!root) {
3600
+ root = document.createElement('div');
3601
+ root.id = 'pdfBody';
3602
+ root.style.position = 'fixed';
3603
+ root.style.left = '-99999px';
3604
+ root.style.top = '0';
3605
+ root.style.width = '794px';
3606
+ root.style.pointerEvents = 'none';
3607
+ document.body.appendChild(root);
3608
+ createdTemp = true;
3609
+ }
3610
+ if (this.additionalFileContents && !root.hasChildNodes()) {
3611
+ const html = String(this.additionalFileContents);
3612
+ if (html.includes('<html')) {
3613
+ const parsed = new DOMParser().parseFromString(html, 'text/html');
3614
+ root.innerHTML = ((_b = parsed.body) === null || _b === void 0 ? void 0 : _b.innerHTML) || html;
3615
+ }
3616
+ else {
3617
+ root.innerHTML = html;
3618
+ }
3619
+ }
3620
+ const allSections = Array.from(root.querySelectorAll('section'));
3621
+ const leafSections = allSections.filter(s => !s.querySelector('section'));
3622
+ const nodesToRender = leafSections.length ? leafSections : (allSections.length ? allSections : [root]);
3623
+ const pageWidth = doc.internal.pageSize.getWidth();
3624
+ const pageHeight = doc.internal.pageSize.getHeight();
3625
+ for (let i = 0; i < nodesToRender.length; i++) {
3626
+ const el = nodesToRender[i];
3583
3627
  try {
3584
- const dataUrl = yield toJpeg(html, {
3585
- quality: 1,
3586
- backgroundColor: 'white'
3587
- });
3588
- const img = new Image();
3589
- img.src = dataUrl;
3628
+ const dataUrl = yield toJpeg(el, { quality: 1, backgroundColor: 'white' });
3590
3629
  yield new Promise((resolve) => {
3630
+ const img = new Image();
3591
3631
  img.onload = () => {
3592
- const imgWidth = 218;
3593
- const imgHeight = (img.height * imgWidth) / img.width;
3594
- doc.addImage(dataUrl, 'JPG', -4, 0, imgWidth, imgHeight);
3595
- if (i < sectionsCount - 1) {
3632
+ const w = img.width;
3633
+ const h = img.height;
3634
+ const scale = Math.min(pageWidth / w, pageHeight / h);
3635
+ const renderW = w * scale;
3636
+ const renderH = h * scale;
3637
+ doc.addImage(dataUrl, 'JPG', 0, 0, renderW, renderH);
3638
+ if (i < nodesToRender.length - 1)
3596
3639
  doc.addPage();
3597
- }
3598
3640
  resolve();
3599
3641
  };
3642
+ img.src = dataUrl;
3600
3643
  });
3601
3644
  }
3602
- catch (error) {
3603
- console.error('Error generating image:', error);
3645
+ catch (err) {
3646
+ console.error('Error generating image:', err);
3604
3647
  }
3605
3648
  }
3606
- const tempFile = doc.output('datauristring');
3607
- const file = this.dataURItoBlob(tempFile);
3608
- const fileAsCoDocument = yield FileUtils.ReadFileAsNewCoDocument(file);
3609
- fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType$1.Image : FileType$1.Document;
3610
- fileAsCoDocument.creationDate = new Date();
3611
- fileAsCoDocument.modifiedDate = new Date();
3612
- fileAsCoDocument.reports = 1;
3613
- this.additionalFileChangeEvent.emit(fileAsCoDocument);
3614
- this.showLoader = false;
3615
- this.closePDFDialog.emit();
3616
- clearButton.hidden = false;
3649
+ if (createdTemp && root && root.parentNode) {
3650
+ root.parentNode.removeChild(root);
3651
+ }
3652
+ // enableLocalPreview can be enabled to see the output without sending email
3653
+ if (this.enableLocalPreview && this._isLocalhost()) {
3654
+ const blob = doc.output('blob');
3655
+ const url = URL.createObjectURL(blob);
3656
+ window.open(url, '_blank');
3657
+ this.showLoader = false;
3658
+ if (clearButton)
3659
+ clearButton.hidden = false;
3660
+ return;
3661
+ }
3662
+ try {
3663
+ const tempFile = doc.output('datauristring');
3664
+ const file = this.dataURItoBlob(tempFile);
3665
+ const fileAsCoDocument = yield FileUtils.ReadFileAsNewCoDocument(file);
3666
+ fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType$1.Image : FileType$1.Document;
3667
+ fileAsCoDocument.creationDate = new Date();
3668
+ fileAsCoDocument.modifiedDate = new Date();
3669
+ fileAsCoDocument.reports = 1;
3670
+ this.additionalFileChangeEvent.emit(fileAsCoDocument);
3671
+ }
3672
+ catch (e) {
3673
+ console.error('PDF packaging failed:', e);
3674
+ }
3675
+ finally {
3676
+ this.showLoader = false;
3677
+ this.closePDFDialog.emit();
3678
+ if (clearButton)
3679
+ clearButton.hidden = false;
3680
+ }
3617
3681
  });
3618
3682
  }
3619
3683
  dataURItoBlob(dataURI) {
@@ -3651,7 +3715,7 @@ CustomPdfDialogComponent.decorators = [
3651
3715
  <div *ngIf="showLoader" class="loader-container">
3652
3716
  <co-loader *ngIf="showLoader"></co-loader>
3653
3717
  </div>
3654
-
3718
+
3655
3719
  <div #pdfBody id="pdfBody">
3656
3720
  <section [innerHTML]="fileBody"></section>
3657
3721
  </div>