UNPKG

ngx-extended-pdf-viewer

Version:

Embedding PDF files in your Angular application. Highly configurable viewer including the toolbar, sidebar, and all the features you're used to.

360 lines 55.5 kB
import { EventEmitter } from '@angular/core'; export class NgxFormSupport { /** Maps the internal ids of the annotations of pdf.js to their field name */ formIdToFullFieldName = {}; formIdToField = {}; radioButtons = {}; formData = {}; initialFormDataStoredInThePDF = {}; formDataChange = new EventEmitter(); PDFViewerApplication; ngZone; // set during the initializaion of the PDF viewer cdr; // set during the initializaion of the PDF viewer reset() { this.formData = {}; this.formIdToFullFieldName = {}; } registerFormSupportWithPdfjs(PDFViewerApplication) { this.PDFViewerApplication = PDFViewerApplication; globalThis.getFormValueFromAngular = (key) => this.getFormValueFromAngular(key); globalThis.updateAngularFormValue = (key, value) => this.updateAngularFormValueCalledByPdfjs(key, value); globalThis.registerAcroformField = (id, element, value, radioButtonValueName, initialValueFromPDF) => this.registerAcroformField(id, element, value, radioButtonValueName, initialValueFromPDF); globalThis.registerXFAField = (element, value, initialValueFromPDF) => this.registerXFAField(element, value, initialValueFromPDF); } registerAcroformField(id, element, value, radioButtonValueName, initialFormValueFromPDF) { const fieldName = element.name; this.formIdToField[id] = element; this.formIdToFullFieldName[id] = fieldName; if (element instanceof HTMLInputElement && element.type === 'radio') { const groupName = fieldName; this.formIdToFullFieldName[id] = groupName; if (value) { this.formData[groupName] = radioButtonValueName; this.initialFormDataStoredInThePDF[groupName] = initialFormValueFromPDF; } element.setAttribute('exportValue', radioButtonValueName); if (!this.radioButtons[groupName]) { this.radioButtons[groupName] = []; } this.radioButtons[groupName].push(element); } else if (element instanceof HTMLSelectElement) { this.formData[fieldName] = this.getValueOfASelectField(element); this.initialFormDataStoredInThePDF[fieldName] = initialFormValueFromPDF; } else { if (value !== undefined) { this.formData[fieldName] = value; } this.initialFormDataStoredInThePDF[fieldName] = initialFormValueFromPDF; } } registerXFAField(element, value, initialFormValueFromPDF) { const fullFieldName = this.findFullXFAName(element); if (element instanceof HTMLInputElement && element.type === 'radio') { const id = element.getAttribute('fieldid') ?? ''; // remove the xfa name of the radio button itself form the field name, // because the field name refers to the entire group of relatated radio buttons const groupName = fullFieldName.substring(0, fullFieldName.lastIndexOf('.')); this.formIdToFullFieldName[id] = groupName; this.formData[groupName] = value?.value; this.initialFormDataStoredInThePDF[groupName] = initialFormValueFromPDF; if (!this.radioButtons[groupName]) { this.radioButtons[groupName] = []; } this.radioButtons[groupName].push(element); } else if (element instanceof HTMLInputElement) { const id = element.getAttribute('fieldid') ?? ''; this.formIdToField[id] = element; this.formIdToFullFieldName[id] = fullFieldName; this.formData[fullFieldName] = value?.value; this.initialFormDataStoredInThePDF[fullFieldName] = initialFormValueFromPDF; } else if (element instanceof HTMLSelectElement) { const id = element.getAttribute('fieldid') ?? ''; this.formIdToField[id] = element; this.formIdToFullFieldName[id] = fullFieldName; this.formData[fullFieldName] = value?.value; this.initialFormDataStoredInThePDF[fullFieldName] = initialFormValueFromPDF; } else if (element instanceof HTMLTextAreaElement) { const id = element.getAttribute('fieldid') ?? ''; this.formIdToField[id] = element; this.formIdToFullFieldName[id] = fullFieldName; this.formData[fullFieldName] = value?.value; this.initialFormDataStoredInThePDF[fullFieldName] = initialFormValueFromPDF; } else { console.error("Couldn't register an XFA form field", element); } } getValueOfASelectField(selectElement) { const { options, multiple } = selectElement; if (!multiple) { return options.selectedIndex === -1 ? null : options[options.selectedIndex]['value']; } return Array.prototype.filter.call(options, (option) => option.selected).map((option) => option['value']); } getFormValueFromAngular(element) { let key; if (element instanceof HTMLElement) { const fieldName = this.findXFAName(element); if (fieldName) { if (this.formData.hasOwnProperty(fieldName)) { key = fieldName; } else { key = this.findFullXFAName(element); } } else { console.error("Couldn't find the field name or XFA name of the form field", element); return { value: null }; } } else { key = element; } return { value: this.formData[key] }; } findXFAName(element) { let parentElement = element; while (!parentElement.getAttribute('xfaname') && parentElement.parentElement) { parentElement = parentElement.parentElement; } if (element instanceof HTMLInputElement && element.type === 'radio') { do { parentElement = parentElement?.parentElement; } while (!parentElement?.getAttribute('xfaname') && parentElement); } let fieldName = parentElement?.getAttribute('xfaname'); if (!fieldName) { throw new Error("Couldn't find the xfaname of the field"); } return fieldName; } findFullXFAName(element) { let parentElement = element; let fieldName = ''; while (parentElement instanceof HTMLElement && parentElement.parentElement) { const xfaName = parentElement.getAttribute('xfaname'); if (xfaName) { fieldName = xfaName + '.' + fieldName; } parentElement = parentElement.parentElement; } if (!fieldName) { throw new Error("Couldn't find the xfaname of the field"); } fieldName = fieldName.substring(0, fieldName.length - 1); if (element instanceof HTMLInputElement && element.type === 'radio') { // ignore the last part of the xfaName because it's actually the value of the field return fieldName.substring(0, fieldName.lastIndexOf('.')); } return fieldName; } updateAngularFormValueCalledByPdfjs(key, value) { if (!this.formData) { this.formData = {}; } // Ignore formattedValue-only updates to prevent clearing Angular form data // The formattedValue is just for display formatting, not actual user input if (value.formattedValue !== undefined && value.value === undefined) { return; } if (typeof key === 'string') { const acroFormKey = this.formIdToFullFieldName[key]; const fullKey = acroFormKey ?? Object.values(this.formIdToFullFieldName).find((k) => k === key || k.endsWith('.' + key)); if (fullKey) { const field = this.formIdToField[key]; let change = this.doUpdateAngularFormValue(field, value, fullKey); if (change) { this.ngZone.run(() => { this.formDataChange.emit(this.formData); this.cdr.detectChanges(); }); } } else { console.error("Couldn't find the field with the name " + key); } } else { let change = false; const shortFieldName = this.findXFAName(key); if (this.formData.hasOwnProperty(shortFieldName)) { change = this.doUpdateAngularFormValue(key, value, shortFieldName); } const fullFieldName = this.findFullXFAName(key); if (fullFieldName !== shortFieldName) { change ||= this.doUpdateAngularFormValue(key, value, fullFieldName); } if (change) { this.ngZone.run(() => { this.formDataChange.emit(this.formData); this.cdr.detectChanges(); }); } } } doUpdateAngularFormValue(field, value, fullKey) { let change = false; // Use the actual user input value, not the formatted display value const actualValue = value.value; if (field instanceof HTMLInputElement && field.type === 'checkbox') { const exportValue = field.getAttribute('exportvalue'); if (exportValue) { if (actualValue) { if (this.formData[fullKey] !== exportValue) { this.formData[fullKey] = exportValue; change = true; } } else { if (this.formData[fullKey] !== false) { this.formData[fullKey] = false; change = true; } } } else if (this.formData[fullKey] !== actualValue) { this.formData[fullKey] = actualValue ?? ''; change = true; } } else if (field instanceof HTMLInputElement && field.type === 'radio') { const exportValue = field.getAttribute('exportvalue') ?? field.getAttribute('xfaon'); if (actualValue) { if (this.formData[fullKey] !== exportValue) { this.formData[fullKey] = exportValue; change = true; } } } else if (this.formData[fullKey] !== actualValue) { this.formData[fullKey] = actualValue ?? ''; change = true; } return change; } updateFormFieldsInPdfCalledByNgOnChanges(previousFormData) { if (!this.PDFViewerApplication?.pdfDocument?.annotationStorage) { // ngOnChanges calls this method too early - so just ignore it return; } for (const key in this.formData) { if (this.formData.hasOwnProperty(key)) { const newValue = this.formData[key]; if (newValue !== previousFormData[key]) { this.setFieldValueAndUpdateAnnotationStorage(key, newValue); } } } for (const key in previousFormData) { if (previousFormData.hasOwnProperty(key) && previousFormData[key]) { let hasPreviousValue = this.formData.hasOwnProperty(key); if (!hasPreviousValue) { const fullKey = Object.keys(this.formData).find((k) => k === key || k.endsWith('.' + key)); if (fullKey) { hasPreviousValue = this.formData.hasOwnProperty(fullKey); } } if (!hasPreviousValue) { this.setFieldValueAndUpdateAnnotationStorage(key, null); } } } // #2691 modified by ngx-extended-pdf-viewer // After programmatically setting form values, update the baseline for change detection if (this.PDFViewerApplication?.setInitialAnnotationValues) { setTimeout(() => { this.PDFViewerApplication?.setInitialAnnotationValues?.(); }, 10); // Small delay to ensure all form updates are processed } // #2691 end of modification by ngx-extended-pdf-viewer } setFieldValueAndUpdateAnnotationStorage(key, newValue) { const radios = this.findRadioButtonGroup(key); if (radios) { radios.forEach((r) => { const activeValue = r.getAttribute('exportValue') ?? r.getAttribute('xfaon'); r.checked = activeValue === newValue; }); const updateFromAngular = new CustomEvent('updateFromAngular', { detail: newValue, }); radios[0].dispatchEvent(updateFromAngular); } else { const fieldIds = this.findFormIdsFromFieldName(key); if (fieldIds) { fieldIds.forEach((fieldId) => { const htmlField = this.formIdToField[fieldId]; if (htmlField) { if (htmlField instanceof HTMLInputElement && htmlField.type === 'checkbox') { let activeValue = htmlField.getAttribute('xfaon') ?? htmlField.getAttribute('exportvalue') ?? true; if (newValue === true || newValue === false) { activeValue = true; } htmlField.checked = activeValue === newValue; } else if (htmlField instanceof HTMLSelectElement) { this.populateSelectField(htmlField, newValue); } else { // textareas and input fields htmlField.value = newValue; } const updateFromAngular = new CustomEvent('updateFromAngular', { detail: newValue, }); htmlField.dispatchEvent(updateFromAngular); } else { console.error("Couldn't set the value of the field", key); } }); } } } populateSelectField(htmlField, newValue) { if (htmlField.multiple) { const { options } = htmlField; const newValueArray = newValue; for (let i = 0; i < options.length; i++) { const option = options.item(i); if (option) { option.selected = newValueArray.some((o) => o === option.value); } } } else { htmlField.value = newValue; } } findFormIdsFromFieldName(fieldName) { if (Object.entries(this.formIdToFullFieldName).length === 0) { // sometimes, ngOnChanges() is called before initializing the PDF file return undefined; } const matchingEntries = Object.entries(this.formIdToFullFieldName).filter((entry) => entry[1] === fieldName || entry[1].endsWith('.' + fieldName)); if (matchingEntries.length === 0) { console.log("Couldn't find the field " + fieldName); return undefined; } return matchingEntries.map((e) => e[0]); } findRadioButtonGroup(fieldName) { const matchingEntries = Object.entries(this.radioButtons).filter((entry) => entry[0].endsWith('.' + fieldName) || entry[0] === fieldName); if (matchingEntries.length === 0) { return null; } if (matchingEntries.length > 1) { console.log('More than one radio button group name matches this name. Please use the qualified field name', matchingEntries.map((radio) => radio[0])); console.log('ngx-extended-pdf-viewer uses the first matching field (which may not be the topmost field on your PDF form): ' + matchingEntries[0][0]); } return matchingEntries[0][1]; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-form-support.js","sourceRoot":"","sources":["../../../../projects/ngx-extended-pdf-viewer/src/lib/ngx-form-support.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, EventEmitter, NgZone } from '@angular/core';\nimport { FormDataType } from './ngx-extended-pdf-viewer.component';\nimport { IPDFViewerApplication } from './options/pdf-viewer-application';\n\nexport type HtmlFormElement = HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement;\n\nexport class NgxFormSupport {\n  /** Maps the internal ids of the annotations of pdf.js to their field name */\n  private formIdToFullFieldName: { [key: string]: string } = {};\n\n  private formIdToField: { [key: string]: HtmlFormElement } = {};\n\n  private radioButtons: { [key: string]: Array<HTMLInputElement> } = {};\n\n  public formData: FormDataType = {};\n\n  public initialFormDataStoredInThePDF: FormDataType = {};\n\n  public formDataChange = new EventEmitter<FormDataType>();\n\n  private PDFViewerApplication: IPDFViewerApplication | undefined;\n\n  public ngZone!: NgZone; // set during the initializaion of the PDF viewer\n\n  public cdr!: ChangeDetectorRef; // set during the initializaion of the PDF viewer\n\n  public reset() {\n    this.formData = {};\n    this.formIdToFullFieldName = {};\n  }\n\n  public registerFormSupportWithPdfjs(PDFViewerApplication: IPDFViewerApplication): void {\n    this.PDFViewerApplication = PDFViewerApplication;\n    (globalThis as any).getFormValueFromAngular = (key: string) => this.getFormValueFromAngular(key);\n    (globalThis as any).updateAngularFormValue = (key: string | HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement, value: { value: string }) =>\n      this.updateAngularFormValueCalledByPdfjs(key, value);\n    (globalThis as any).registerAcroformField = (\n      id: string,\n      element: HtmlFormElement,\n      value: string | Array<string>,\n      radioButtonValueName: string,\n      initialValueFromPDF: string,\n    ) => this.registerAcroformField(id, element, value, radioButtonValueName, initialValueFromPDF);\n\n    (globalThis as any).registerXFAField = (element: HtmlFormElement, value: { value: string }, initialValueFromPDF: string) =>\n      this.registerXFAField(element, value, initialValueFromPDF);\n  }\n\n  private registerAcroformField(\n    id: string,\n    element: HtmlFormElement,\n    value: null | string | Array<string>,\n    radioButtonValueName: string,\n    initialFormValueFromPDF: string,\n  ): void {\n    const fieldName = element.name;\n    this.formIdToField[id] = element;\n    this.formIdToFullFieldName[id] = fieldName;\n    if (element instanceof HTMLInputElement && element.type === 'radio') {\n      const groupName = fieldName;\n      this.formIdToFullFieldName[id] = groupName;\n      if (value) {\n        this.formData[groupName] = radioButtonValueName;\n        this.initialFormDataStoredInThePDF[groupName] = initialFormValueFromPDF;\n      }\n      element.setAttribute('exportValue', radioButtonValueName);\n      if (!this.radioButtons[groupName]) {\n        this.radioButtons[groupName] = [];\n      }\n      this.radioButtons[groupName].push(element);\n    } else if (element instanceof HTMLSelectElement) {\n      this.formData[fieldName] = this.getValueOfASelectField(element);\n      this.initialFormDataStoredInThePDF[fieldName] = initialFormValueFromPDF;\n    } else {\n      if (value !== undefined) {\n        this.formData[fieldName] = value;\n      }\n      this.initialFormDataStoredInThePDF[fieldName] = initialFormValueFromPDF;\n    }\n  }\n\n  private registerXFAField(element: HTMLElement, value: { value: string }, initialFormValueFromPDF: string): void {\n    const fullFieldName = this.findFullXFAName(element);\n    if (element instanceof HTMLInputElement && element.type === 'radio') {\n      const id = element.getAttribute('fieldid') ?? '';\n      // remove the xfa name of the radio button itself form the field name,\n      // because the field name refers to the entire group of relatated radio buttons\n      const groupName = fullFieldName.substring(0, fullFieldName.lastIndexOf('.'));\n      this.formIdToFullFieldName[id] = groupName;\n      this.formData[groupName] = value?.value;\n      this.initialFormDataStoredInThePDF[groupName] = initialFormValueFromPDF;\n\n      if (!this.radioButtons[groupName]) {\n        this.radioButtons[groupName] = [];\n      }\n      this.radioButtons[groupName].push(element);\n    } else if (element instanceof HTMLInputElement) {\n      const id = element.getAttribute('fieldid') ?? '';\n      this.formIdToField[id] = element;\n      this.formIdToFullFieldName[id] = fullFieldName;\n      this.formData[fullFieldName] = value?.value;\n      this.initialFormDataStoredInThePDF[fullFieldName] = initialFormValueFromPDF;\n    } else if (element instanceof HTMLSelectElement) {\n      const id = element.getAttribute('fieldid') ?? '';\n      this.formIdToField[id] = element;\n      this.formIdToFullFieldName[id] = fullFieldName;\n      this.formData[fullFieldName] = value?.value;\n      this.initialFormDataStoredInThePDF[fullFieldName] = initialFormValueFromPDF;\n    } else if (element instanceof HTMLTextAreaElement) {\n      const id = element.getAttribute('fieldid') ?? '';\n      this.formIdToField[id] = element;\n      this.formIdToFullFieldName[id] = fullFieldName;\n      this.formData[fullFieldName] = value?.value;\n      this.initialFormDataStoredInThePDF[fullFieldName] = initialFormValueFromPDF;\n    } else {\n      console.error(\"Couldn't register an XFA form field\", element);\n    }\n  }\n\n  private getValueOfASelectField(selectElement: HTMLSelectElement): null | string | Array<string> {\n    const { options, multiple } = selectElement;\n    if (!multiple) {\n      return options.selectedIndex === -1 ? null : options[options.selectedIndex]['value'];\n    }\n    return Array.prototype.filter.call(options, (option) => option.selected).map((option) => option['value']);\n  }\n\n  private getFormValueFromAngular(element: HTMLElement | string): Object {\n    let key: string;\n    if (element instanceof HTMLElement) {\n      const fieldName = this.findXFAName(element);\n      if (fieldName) {\n        if (this.formData.hasOwnProperty(fieldName)) {\n          key = fieldName;\n        } else {\n          key = this.findFullXFAName(element);\n        }\n      } else {\n        console.error(\"Couldn't find the field name or XFA name of the form field\", element);\n        return { value: null };\n      }\n    } else {\n      key = element;\n    }\n    return { value: this.formData[key] };\n  }\n\n  private findXFAName(element: HTMLElement): string {\n    let parentElement: HTMLElement | null | undefined = element;\n    while (!parentElement.getAttribute('xfaname') && parentElement.parentElement) {\n      parentElement = parentElement.parentElement;\n    }\n    if (element instanceof HTMLInputElement && element.type === 'radio') {\n      do {\n        parentElement = parentElement?.parentElement;\n      } while (!parentElement?.getAttribute('xfaname') && parentElement);\n    }\n    let fieldName = parentElement?.getAttribute('xfaname');\n    if (!fieldName) {\n      throw new Error(\"Couldn't find the xfaname of the field\");\n    }\n    return fieldName;\n  }\n\n  private findFullXFAName(element: HTMLElement): string {\n    let parentElement = element;\n    let fieldName = '';\n    while (parentElement instanceof HTMLElement && parentElement.parentElement) {\n      const xfaName = parentElement.getAttribute('xfaname');\n      if (xfaName) {\n        fieldName = xfaName + '.' + fieldName;\n      }\n      parentElement = parentElement.parentElement;\n    }\n    if (!fieldName) {\n      throw new Error(\"Couldn't find the xfaname of the field\");\n    }\n    fieldName = fieldName.substring(0, fieldName.length - 1);\n    if (element instanceof HTMLInputElement && element.type === 'radio') {\n      // ignore the last part of the xfaName because it's actually the value of the field\n      return fieldName.substring(0, fieldName.lastIndexOf('.'));\n    }\n    return fieldName;\n  }\n\n  private updateAngularFormValueCalledByPdfjs(\n    key: string | HTMLSelectElement | HTMLInputElement | HTMLTextAreaElement,\n    value: { value?: string; formattedValue?: string },\n  ): void {\n    if (!this.formData) {\n      this.formData = {};\n    }\n\n    // Ignore formattedValue-only updates to prevent clearing Angular form data\n    // The formattedValue is just for display formatting, not actual user input\n    if (value.formattedValue !== undefined && value.value === undefined) {\n      return;\n    }\n\n    if (typeof key === 'string') {\n      const acroFormKey = this.formIdToFullFieldName[key];\n      const fullKey = acroFormKey ?? Object.values(this.formIdToFullFieldName).find((k) => k === key || k.endsWith('.' + key));\n      if (fullKey) {\n        const field = this.formIdToField[key];\n        let change = this.doUpdateAngularFormValue(field, value, fullKey);\n        if (change) {\n          this.ngZone.run(() => {\n            this.formDataChange.emit(this.formData);\n            this.cdr.detectChanges();\n          });\n        }\n      } else {\n        console.error(\"Couldn't find the field with the name \" + key);\n      }\n    } else {\n      let change = false;\n      const shortFieldName = this.findXFAName(key);\n      if (this.formData.hasOwnProperty(shortFieldName)) {\n        change = this.doUpdateAngularFormValue(key, value, shortFieldName);\n      }\n      const fullFieldName = this.findFullXFAName(key);\n      if (fullFieldName !== shortFieldName) {\n        change ||= this.doUpdateAngularFormValue(key, value, fullFieldName);\n      }\n      if (change) {\n        this.ngZone.run(() => {\n          this.formDataChange.emit(this.formData);\n          this.cdr.detectChanges();\n        });\n      }\n    }\n  }\n\n  private doUpdateAngularFormValue(field: HtmlFormElement, value: { value?: string; formattedValue?: string }, fullKey: string) {\n    let change = false;\n    // Use the actual user input value, not the formatted display value\n    const actualValue = value.value;\n    if (field instanceof HTMLInputElement && field.type === 'checkbox') {\n      const exportValue = field.getAttribute('exportvalue');\n      if (exportValue) {\n        if (actualValue) {\n          if (this.formData[fullKey] !== exportValue) {\n            this.formData[fullKey] = exportValue;\n            change = true;\n          }\n        } else {\n          if (this.formData[fullKey] !== false) {\n            this.formData[fullKey] = false;\n            change = true;\n          }\n        }\n      } else if (this.formData[fullKey] !== actualValue) {\n        this.formData[fullKey] = actualValue ?? '';\n        change = true;\n      }\n    } else if (field instanceof HTMLInputElement && field.type === 'radio') {\n      const exportValue = field.getAttribute('exportvalue') ?? field.getAttribute('xfaon');\n      if (actualValue) {\n        if (this.formData[fullKey] !== exportValue) {\n          this.formData[fullKey] = exportValue;\n          change = true;\n        }\n      }\n    } else if (this.formData[fullKey] !== actualValue) {\n      this.formData[fullKey] = actualValue ?? '';\n      change = true;\n    }\n    return change;\n  }\n\n  public updateFormFieldsInPdfCalledByNgOnChanges(previousFormData: Object) {\n    if (!this.PDFViewerApplication?.pdfDocument?.annotationStorage) {\n      // ngOnChanges calls this method too early - so just ignore it\n      return;\n    }\n\n    for (const key in this.formData) {\n      if (this.formData.hasOwnProperty(key)) {\n        const newValue = this.formData[key];\n        if (newValue !== previousFormData[key]) {\n          this.setFieldValueAndUpdateAnnotationStorage(key, newValue);\n        }\n      }\n    }\n\n    for (const key in previousFormData) {\n      if (previousFormData.hasOwnProperty(key) && previousFormData[key]) {\n        let hasPreviousValue = this.formData.hasOwnProperty(key);\n        if (!hasPreviousValue) {\n          const fullKey = Object.keys(this.formData).find((k) => k === key || k.endsWith('.' + key));\n          if (fullKey) {\n            hasPreviousValue = this.f