Angular Storybook 中的 FormGroup:将循环结构转换为 JSON
Angular FormGroup in Storybook: Converting circular structure to JSON
我正在使用 angular 和故事书。我的模型中有 FormGroup 和 FormArray,但它们无法使用故事书。
a.stories.ts ->
import { CommonModule } from '@angular/common';
import { FormArray, FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';
import { Meta, moduleMetadata, Story } from '@storybook/angular';
export default {
title: 'Example/A',
component: AComponent,
decorators: [
moduleMetadata({
imports: [
CommonModule,
ReactiveFormsModule,
],
providers: [],
}),
],
} as Meta;
const Template: Story<AComponent> = (args: AComponent) => ({
props: args,
});
export const Costs = Template.bind({});
Costs.args = {
model: {
questions: [
{
...,
"question": "lorem ipsum",
"formGroup": new FormGroup({
answers: new FormArray([
new FormGroup({
"Q020_A001": new FormControl(null, [Validators.required]),
"Q020_A002": new FormControl(null, [Validators.required]),
}),
]),
}),
"answers": [
{
...,
"answerCode": "Q020_A001",
},
{
...,
"answerCode": "Q020_A002",
},
],
}
],
},
};
我在故事书中遇到错误 ->
TypeError: Converting circular structure to JSON
--> starting at object with constructor 'FormGroup'
| property 'controls' -> object with constructor 'Object'
| property 'answers' -> object with constructor 'FormArray'
--- property '_parent' closes the circle
at JSON.stringify (<anonymous>)
如果“formGroup”为空则有效。 ->
"formGroup": new FormGroup({
}),
但如果“formGroup”不为空,则无法使用。 ->
"formGroup": new FormGroup({
asd: new FormControl(''),
}),
我该如何解决这个问题?
问题是您提供给 Storybook 的每个 args
,SB 都会调用 JSON.stringify
。这种设计是必需的,因为 SB 也在运行时使用你的 args
并允许你更改值。
在您的情况下,您创建了一个 Reactive 表单模型,导致对象无法通过 JSON.stringify
转换为字符串。
要修复它,您需要:
- 仅将参数作为原始数据提供给 SB
- 更改
AComponent
中的逻辑以根据上面提供的数据创建模型
示例代码:
/// a.stories.ts
Costs.args = {
model: {
questions: [
{
...,
"question": "lorem ipsum",
"formGroup": {
answers: [
{
"Q020_A001": null,
"Q020_A002": null,
}),
]),
},
"answers": [
{
...,
"answerCode": "Q020_A001",
},
{
...,
"answerCode": "Q020_A002",
},
],
}
],
},
};
/// A.component.ts
export class AComponent {
@Input() model;
constructor() {
// create your reactive forms here with data from model
this.form = new FormGroup(...)
}
}
我通过创建一个新的 AStoryComponent 解决了这个问题。
// a-story.component.ts
@Component({
selector: 'a-component-story',
template: `<a-component [model]="model"></a-component>`,
})
export class AStoryComponent implements OnInit {
@Input() model!: any;
constructor() {}
ngOnInit() {
// Converts the "formGroup" property I specified in a.stories.ts to the real formGroup object.
convertStoryModelToRealModel(this.model);
}
}
// a.stories.ts
import { Validators } from '@angular/forms';
import { Meta, moduleMetadata, Story } from '@storybook/angular';
export default {
title: 'Example/A',
component: AStoryComponent,
decorators: [
moduleMetadata({
declarations: [AStoryComponent, AComponent],
imports: [
CommonModule,
ReactiveFormsModule,
],
}),
],
} as Meta;
const Template: Story<AStoryComponent> = (args: AStoryComponent) => ({
props: args,
});
export const Costs = Template.bind({});
Costs.args = {
model: {
questions: [
{
...,
"question": "lorem ipsum",
"formGroup": {
answers: [
{
"Q020_A001": [null, [Validators.required]],
"Q020_A002": [null, [Validators.required]],
},
],
},
"answers": [
{
...,
"answerCode": "Q020_A001",
},
{
...,
"answerCode": "Q020_A002",
},
],
}
],
} as any,
};
我正在使用 angular 和故事书。我的模型中有 FormGroup 和 FormArray,但它们无法使用故事书。
a.stories.ts ->
import { CommonModule } from '@angular/common';
import { FormArray, FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';
import { Meta, moduleMetadata, Story } from '@storybook/angular';
export default {
title: 'Example/A',
component: AComponent,
decorators: [
moduleMetadata({
imports: [
CommonModule,
ReactiveFormsModule,
],
providers: [],
}),
],
} as Meta;
const Template: Story<AComponent> = (args: AComponent) => ({
props: args,
});
export const Costs = Template.bind({});
Costs.args = {
model: {
questions: [
{
...,
"question": "lorem ipsum",
"formGroup": new FormGroup({
answers: new FormArray([
new FormGroup({
"Q020_A001": new FormControl(null, [Validators.required]),
"Q020_A002": new FormControl(null, [Validators.required]),
}),
]),
}),
"answers": [
{
...,
"answerCode": "Q020_A001",
},
{
...,
"answerCode": "Q020_A002",
},
],
}
],
},
};
我在故事书中遇到错误 ->
TypeError: Converting circular structure to JSON
--> starting at object with constructor 'FormGroup'
| property 'controls' -> object with constructor 'Object'
| property 'answers' -> object with constructor 'FormArray'
--- property '_parent' closes the circle
at JSON.stringify (<anonymous>)
如果“formGroup”为空则有效。 ->
"formGroup": new FormGroup({
}),
但如果“formGroup”不为空,则无法使用。 ->
"formGroup": new FormGroup({
asd: new FormControl(''),
}),
我该如何解决这个问题?
问题是您提供给 Storybook 的每个 args
,SB 都会调用 JSON.stringify
。这种设计是必需的,因为 SB 也在运行时使用你的 args
并允许你更改值。
在您的情况下,您创建了一个 Reactive 表单模型,导致对象无法通过 JSON.stringify
转换为字符串。
要修复它,您需要:
- 仅将参数作为原始数据提供给 SB
- 更改
AComponent
中的逻辑以根据上面提供的数据创建模型
示例代码:
/// a.stories.ts
Costs.args = {
model: {
questions: [
{
...,
"question": "lorem ipsum",
"formGroup": {
answers: [
{
"Q020_A001": null,
"Q020_A002": null,
}),
]),
},
"answers": [
{
...,
"answerCode": "Q020_A001",
},
{
...,
"answerCode": "Q020_A002",
},
],
}
],
},
};
/// A.component.ts
export class AComponent {
@Input() model;
constructor() {
// create your reactive forms here with data from model
this.form = new FormGroup(...)
}
}
我通过创建一个新的 AStoryComponent 解决了这个问题。
// a-story.component.ts
@Component({
selector: 'a-component-story',
template: `<a-component [model]="model"></a-component>`,
})
export class AStoryComponent implements OnInit {
@Input() model!: any;
constructor() {}
ngOnInit() {
// Converts the "formGroup" property I specified in a.stories.ts to the real formGroup object.
convertStoryModelToRealModel(this.model);
}
}
// a.stories.ts
import { Validators } from '@angular/forms';
import { Meta, moduleMetadata, Story } from '@storybook/angular';
export default {
title: 'Example/A',
component: AStoryComponent,
decorators: [
moduleMetadata({
declarations: [AStoryComponent, AComponent],
imports: [
CommonModule,
ReactiveFormsModule,
],
}),
],
} as Meta;
const Template: Story<AStoryComponent> = (args: AStoryComponent) => ({
props: args,
});
export const Costs = Template.bind({});
Costs.args = {
model: {
questions: [
{
...,
"question": "lorem ipsum",
"formGroup": {
answers: [
{
"Q020_A001": [null, [Validators.required]],
"Q020_A002": [null, [Validators.required]],
},
],
},
"answers": [
{
...,
"answerCode": "Q020_A001",
},
{
...,
"answerCode": "Q020_A002",
},
],
}
],
} as any,
};