mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-03 05:04:51 +08:00
"Avoid component redeclaration" re-implementation
This commit is contained in:
parent
8e26189e37
commit
21e4997314
@ -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,
|
||||
|
1
app/angular/index.d.ts
vendored
1
app/angular/index.d.ts
vendored
@ -45,7 +45,6 @@ export interface IStory {
|
||||
props?: ICollection;
|
||||
moduleMetadata?: Partial<NgModuleMetadata>;
|
||||
component?: any;
|
||||
requiresComponentDeclaration?: boolean;
|
||||
template?: string;
|
||||
}
|
||||
|
||||
|
@ -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,
|
||||
|
@ -12,7 +12,6 @@ export interface ICollection {
|
||||
|
||||
export interface NgStory {
|
||||
component?: any;
|
||||
requiresComponentDeclaration?: boolean;
|
||||
props: ICollection;
|
||||
propsMeta?: ICollection;
|
||||
moduleMetadata?: NgModuleMetadata;
|
||||
|
@ -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,
|
||||
};
|
||||
});
|
||||
|
Loading…
x
Reference in New Issue
Block a user