@revolist/angular-datagrid
Version:
Angular DataGrid Spreadsheet component with native cell render support
71 lines • 11.2 kB
JavaScript
/**
* Angular Adapter Function:
* Handles the rendering and lifecycle of Angular components within StencilJS using updated Angular APIs.
*/
import { Injector, ApplicationRef, createComponent, EnvironmentInjector, inject } from '@angular/core';
function updateProps(el, newProps) {
if (!el.componentRef?.instance) {
return;
}
// Update both ways to ensure it works with different component implementations
el.componentRef.instance.props = newProps;
el.componentRef.setInput('props', newProps);
// Handle ngOnChanges if component implements it
if (el.componentRef.instance.ngOnChanges) {
const previousValue = el.componentRef.instance.props;
const simpleChanges = {
props: {
previousValue,
currentValue: newProps,
firstChange: previousValue === undefined,
isFirstChange: () => previousValue === undefined
}
};
el.componentRef.instance.ngOnChanges(simpleChanges);
}
// Force component to check for updates
el.componentRef.changeDetectorRef.markForCheck();
el.componentRef.changeDetectorRef.detectChanges();
}
export function TemplateConstructor(el, AngularComponent, initialProps = {}, injector, lastEl = null) {
if (!el) {
lastEl?.destroy?.(); // Destroy last component if element is null
return null;
}
if (!el.componentRef) {
const applicationRef = injector.get(ApplicationRef);
const environmentInjector = injector.get(EnvironmentInjector);
const componentRef = createComponent(AngularComponent, { environmentInjector });
applicationRef.attachView(componentRef.hostView);
componentRef.instance.props = initialProps;
componentRef.changeDetectorRef.detectChanges();
el.componentRef = componentRef; // Store the component reference
el.appendChild(componentRef.location.nativeElement);
}
else if (el.componentRef.instance) {
updateProps(el, initialProps);
}
// Function to update component props
const update = (newProps) => updateProps(el, newProps);
// Function to destroy the component
const destroy = () => {
el.componentRef?.destroy();
el.componentRef = undefined;
};
return { update, destroy };
}
// Function to create template for Angular component
export const Template = (AngularComponent, customProps, injector = inject(Injector)) => {
return (h, p, addition) => {
const props = customProps ? { ...customProps, ...p } : p;
props.addition = addition;
let lastEl = null;
return h('span', {
key: `${p.prop}-${p.rowIndex || 0}`,
ref: (el) => {
lastEl = TemplateConstructor(el, AngularComponent, props, injector, lastEl);
}
});
};
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"renderer.js","sourceRoot":"","sources":["../../../../projects/angular-datagrid/src/lib/renderer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Angular Adapter Function:\n * Handles the rendering and lifecycle of Angular components within StencilJS using updated Angular APIs.\n */\nimport { Injector, ComponentRef, Type, ApplicationRef, createComponent, EnvironmentInjector, inject } from '@angular/core';\nimport { ColumnDataSchemaModel, ColumnTemplateProp } from '@revolist/revogrid';\n\nexport interface AngularElement extends HTMLElement {\n  componentRef?: ComponentRef<any>;  // Reference to the Angular component\n}\n\nexport interface RenderedComponent<T> {\n  update: (newProps: T) => void;     // Function to update component with new props\n  destroy: () => void;               // Function to destroy the component\n}\n\nfunction updateProps<T extends object>(el: AngularElement, newProps: T) {\n  if (!el.componentRef?.instance) {\n    return;\n  }\n  \n  \n  // Update both ways to ensure it works with different component implementations\n  el.componentRef.instance.props = newProps;\n  el.componentRef.setInput('props', newProps);\n  \n  // Handle ngOnChanges if component implements it\n  if (el.componentRef.instance.ngOnChanges) {\n    const previousValue = el.componentRef.instance.props;\n    const simpleChanges = {\n      props: {\n        previousValue,\n        currentValue: newProps,\n        firstChange: previousValue === undefined,\n        isFirstChange: () => previousValue === undefined\n      }\n    };\n    el.componentRef.instance.ngOnChanges(simpleChanges);\n  }\n  \n  // Force component to check for updates\n  el.componentRef.changeDetectorRef.markForCheck();\n  el.componentRef.changeDetectorRef.detectChanges();\n}\n\nexport function TemplateConstructor<T extends Object>(\n  el: AngularElement | null,\n  AngularComponent: Type<any>,\n  initialProps: T = {} as T,\n  injector: Injector,\n  lastEl: RenderedComponent<T> | null = null,\n): RenderedComponent<T> | null {\n  if (!el) {\n    lastEl?.destroy?.();  // Destroy last component if element is null\n    return null;\n  }\n\n  if (!el.componentRef) {\n    const applicationRef = injector.get(ApplicationRef);\n    const environmentInjector = injector.get(EnvironmentInjector);\n    const componentRef = createComponent(AngularComponent, { environmentInjector });\n    applicationRef.attachView(componentRef.hostView);\n    componentRef.instance.props = initialProps;\n    componentRef.changeDetectorRef.detectChanges();\n    el.componentRef = componentRef;  // Store the component reference\n    el.appendChild(componentRef.location.nativeElement);\n  } else if (el.componentRef.instance) {\n    updateProps(el, initialProps);\n  }\n\n  // Function to update component props\n  const update = (newProps: T) => updateProps(el, newProps);\n\n  // Function to destroy the component\n  const destroy = () => {\n    el.componentRef?.destroy();\n    el.componentRef = undefined;\n  };\n\n  return { update, destroy };\n}\n\n// Function to create template for Angular component\nexport const Template = (\n  AngularComponent: Type<any>,\n  customProps?: any,\n  injector = inject(Injector)\n) => {\n  return (h: any, p: ColumnDataSchemaModel | ColumnTemplateProp, addition?: any) => {\n    const props = customProps ? { ...customProps, ...p } : p;\n    props.addition = addition;\n    let lastEl: RenderedComponent<any> | null = null;\n    return h('span', {\n      key: `${p.prop}-${p.rowIndex || 0}`,\n      ref: (el: AngularElement | null) => {\n        lastEl = TemplateConstructor(el, AngularComponent, props, injector, lastEl);\n      }\n    });\n  };\n};\n"]}