gpt4 book ai didi

具有动态嵌套组件的 Angular 循环依赖项

转载 作者:行者123 更新时间:2023-12-04 15:24:30 27 4
gpt4 key购买 nike

我已经在互联网上寻找解决方案无济于事(不同的情况等)所以请原谅下面的代码转储,我遇到的问题与循环依赖有关。代码转储是为了提供上下文。
旁注:我对 Angular 和 Typescript 还很陌生。
这个概念
当然,我正在尝试构建一组嵌套组件,这些组件都扩展了基类以简化编码。这些组件可以包含彼此的任何子组件。为了加载子组件,基类使用一个指令来决定在它的位置加载哪个组件。最好的例子就是考虑嵌套<div><section>块。
这是我的代码:
指令 directive.tsload() 时加载其各自的组件叫做

import { ComponentFactoryResolver, Directive, ViewContainerRef } from '@angular/core';

import { DivComponent } from './div'; // < -- Part of the dependency issues
import { SectionComponent } from './section'; // < -- Part of the dependency issues

@Directive({
selector: '[child-node]',
})
export class ChildNodeDirective {

@Input() type: string;
@Input() content: any;

constructor(
private container: ViewContainerRef,
private resolver: ComponentFactoryResolver,
){}

load(): void {

let component: Type<any>;

switch (this.type) {
case 'div' : component = DivComponent; break;
case 'section' : component = SectionComponent; break;
}

if (component) {
const factory = this.resolver.resolveComponentFactory(component);
const componentRef = this.container.createComponent(factory);
componentRef.instance.content = this.content;
}

}

}
基类 base.ts此类是所有组件的基础;
  • 用途 @ViewChildrenngAfterContentChecked加载其子指令,
  • childNodesset content被调用以便组件可以呈现其 <child-node>元素
  • import { AfterContentChecked, QueryList, ViewChildren } from '@angular/core';

    import { ChildNodeDirective } from './directive'; // < -- Part of the dependency issues

    export abstract class Base implements AfterContentChecked {

    // These are elements that the template will render into the directive
    public childNodes: {[type: string]: any} = {};

    private childrenLoaded = false;

    @ViewChildren(ChildNodeDirective) protected children?: QueryList<any>;

    ngAfterContentChecked(): void {
    if (!this.childrenLoaded && this.children) {
    this.children.forEach((child: ChildNodeDirective) => {
    child.load();
    });
    this.childrenLoaded = true;
    }
    }

    set content(content: any) {
    this.childNodes = content.childNodes;
    }

    }
    div 组件 div.ts该组件扩展了基础并简单地呈现其子节点
    import { Component } from '@angular/core';
    import { Base } from './base'; // < -- Part of the dependency issues

    @Component({
    selector: 'custom-div',
    templateUrl: './div.html',
    })
    export class DivComponent extends Base {

    public textContent: string;

    set content(content: any) {

    super.content = content;

    this.textContent = content.text;

    }

    }
    div模板 div.html

    <div>{{ textContent }}</div>

    <div *ngFor="let child of childNodes">
    <ng-template child-node [content]="child.content" [type]="child.type"></ng-template>
    </div>


    TL; DR 问题
    所有这些似乎都有效。我能够制作各种内容和 child 的深层嵌套等。我无法谈论代码/实现的“正确性”,但我遇到的唯一问题是循环依赖警告。
    WARNING in Circular dependency detected:
    div.ts -> base.ts -> directive.ts -> div.ts

    WARNING in Circular dependency detected:
    section.ts -> base.ts -> directive.ts -> section.ts
    请帮我摆脱它们...
    解决方案
    根据 Kai 的最后一个建议,我创建了一个装饰器来收集在指令中使用的元数据。 directive.ts的变化

    export const HtmlElementMap: {component: Type<any>, map: string[]}[] = [];

    export function HtmlComponent(config: {map: string[]}) {
    return (target: Type<any>) => {
    ElementMap.push({component: target, map: config.map});
    };
    }

    @Directive({
    selector: '[child-node]',
    })
    export class ChildNodeDirective {

    ...

    load(): void {

    let component: Type<any>;

    HtmlElementMap
    .filter(v => v.map.indexOf(this.type) > -1)
    .forEach(
    v => {
    if (undefined === component) {
    component = v.component;
    }
    }
    );

    if (component) {
    const factory = this.resolver.resolveComponentFactory(component);
    const componentRef = this.container.createComponent(factory);
    componentRef.instance.content = this.content;
    }

    }

    }

    div.ts以及后续组件

    @HtmlComponent({map: ['div']})
    @Component({
    selector: 'custom-div',
    templateUrl: './div.html',
    })
    export class DivComponent extends Base {

    .
    .
    .

    }

    最佳答案

    在我看来,我的图表中的红线是这里的问题。因此,如果您在运行时注册您的组件(需要 switch 语句),您可以尝试解决并删除此依赖项。例如 Base类可以收集所有可用的组件并将它们传递给 ChildNodeDirective类(class)。或者您可以使用服务来注册它们。如果您有可用的 stackblitz 示例,我们可以帮助您编写代码。
    enter image description here

    关于具有动态嵌套组件的 Angular 循环依赖项,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62570795/

    27 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com