"Avoid component redeclaration" re-implementation

This commit is contained in:
Max 2019-04-28 01:17:27 +10:00
parent 8e26189e37
commit 21e4997314
5 changed files with 120 additions and 35 deletions

View File

@ -1,5 +1,5 @@
/* eslint-disable import/no-extraneous-dependencies */
import { Component, Type } from '@angular/core';
import { Component, Type, NgModule } from '@angular/core';
import { FormsModule } from '@angular/forms';
import { BrowserModule } from '@angular/platform-browser';
import { AppComponent } from './app.component';
@ -30,15 +30,55 @@ const createComponentFromTemplate = (template: string) => {
template,
})(componentClass);
};
const extractNgModuleMetadata = (importItem: any): NgModule => {
const decoratorKey = '__annotations__';
const decorators: any[] =
Reflect && Reflect.getOwnPropertyDescriptor
? Reflect.getOwnPropertyDescriptor(importItem, decoratorKey).value
: importItem[decoratorKey];
if (!decorators || decorators.length === 0) {
return null;
}
const ngModuleDecorator: NgModule | undefined = decorators.find(
decorator => decorator instanceof NgModule
);
if (!ngModuleDecorator) {
return null;
}
return ngModuleDecorator;
};
const getExistenceOfComponentInModules = (
component: any,
declarations: any[],
imports: any[]
): boolean => {
if (declarations && declarations.some(declaration => declaration === component)) {
// Found component in declarations array
return true;
}
if (!imports) {
return false;
}
return imports.some(importItem => {
const extractedNgModuleMetadata = extractNgModuleMetadata(importItem);
if (!extractedNgModuleMetadata) {
// Not an NgModule
return false;
}
return getExistenceOfComponentInModules(
component,
extractedNgModuleMetadata.declarations,
extractedNgModuleMetadata.imports
);
});
};
export const initModuleData = (storyObj: NgStory): any => {
const {
component,
template,
props,
moduleMetadata = {},
requiresComponentDeclaration = true,
} = storyObj;
const { component, template, props, moduleMetadata = {} } = storyObj;
const isCreatingComponentFromTemplate = Boolean(template);
@ -46,10 +86,17 @@ export const initModuleData = (storyObj: NgStory): any => {
? createComponentFromTemplate(template)
: component;
const componentDeclarations =
isCreatingComponentFromTemplate || requiresComponentDeclaration
? [AppComponent, AnnotatedComponent]
: [AppComponent];
const componentRequiesDeclaration =
isCreatingComponentFromTemplate ||
!getExistenceOfComponentInModules(
component,
moduleMetadata.declarations,
moduleMetadata.imports
);
const componentDeclarations = componentRequiesDeclaration
? [AppComponent, AnnotatedComponent]
: [AppComponent];
const story = {
component: AnnotatedComponent,

View File

@ -45,7 +45,6 @@ export interface IStory {
props?: ICollection;
moduleMetadata?: Partial<NgModuleMetadata>;
component?: any;
requiresComponentDeclaration?: boolean;
template?: string;
}

View File

@ -47,16 +47,56 @@ const createComponentFromTemplate = (template: string, styles: string[]) => {
})(componentClass);
};
const extractNgModuleMetadata = (importItem: any): NgModule => {
const decoratorKey = '__annotations__';
const decorators: any[] =
Reflect && Reflect.getOwnPropertyDescriptor
? Reflect.getOwnPropertyDescriptor(importItem, decoratorKey).value
: importItem[decoratorKey];
if (!decorators || decorators.length === 0) {
return null;
}
const ngModuleDecorator: NgModule | undefined = decorators.find(
decorator => decorator instanceof NgModule
);
if (!ngModuleDecorator) {
return null;
}
return ngModuleDecorator;
};
const getExistenceOfComponentInModules = (
component: any,
declarations: any[],
imports: any[]
): boolean => {
if (declarations && declarations.some(declaration => declaration === component)) {
// Found component in declarations array
return true;
}
if (!imports) {
return false;
}
return imports.some(importItem => {
const extractedNgModuleMetadata = extractNgModuleMetadata(importItem);
if (!extractedNgModuleMetadata) {
// Not an NgModule
return false;
}
return getExistenceOfComponentInModules(
component,
extractedNgModuleMetadata.declarations,
extractedNgModuleMetadata.imports
);
});
};
const initModule = (storyFn: IStoryFn) => {
const storyObj = storyFn();
const {
component,
template,
props,
styles,
moduleMetadata = {},
requiresComponentDeclaration = true,
} = storyObj;
const { component, template, props, styles, moduleMetadata = {} } = storyObj;
const isCreatingComponentFromTemplate = Boolean(template);
@ -64,10 +104,17 @@ const initModule = (storyFn: IStoryFn) => {
? createComponentFromTemplate(template, styles)
: component;
const componentDeclarations =
isCreatingComponentFromTemplate || requiresComponentDeclaration
? [AppComponent, AnnotatedComponent]
: [AppComponent];
const componentRequiesDeclaration =
isCreatingComponentFromTemplate ||
!getExistenceOfComponentInModules(
component,
moduleMetadata.declarations,
moduleMetadata.imports
);
const componentDeclarations = componentRequiesDeclaration
? [AppComponent, AnnotatedComponent]
: [AppComponent];
const story = {
component: AnnotatedComponent,

View File

@ -12,7 +12,6 @@ export interface ICollection {
export interface NgStory {
component?: any;
requiresComponentDeclaration?: boolean;
props: ICollection;
propsMeta?: ICollection;
moduleMetadata?: NgModuleMetadata;

View File

@ -33,17 +33,12 @@ storiesOf('Custom|Feature Module as Context', module)
};
return {
component: ChipsGroupComponent,
requiresComponentDeclaration: false,
props,
};
},
{
notes: `
This component includes a child component, a pipe, and a default provider, all which come from
the specified feature module.
This behavior is possible by setting the "requiresComponentDeclaration" flag to false.
`.replace(/ {1,}/g, ' '),
notes: `This component includes a child component, a pipe, and a default provider, all which come from
the specified feature module.`,
}
)
.add('Component with default providers', () => {
@ -53,7 +48,6 @@ storiesOf('Custom|Feature Module as Context', module)
};
return {
component: ChipComponent,
requiresComponentDeclaration: false,
props,
};
})
@ -72,7 +66,6 @@ storiesOf('Custom|Feature Module as Context', module)
},
],
},
requiresComponentDeclaration: false,
props,
};
});