我不知道如何添加到我的组件<component></component>一个动态类属性,但在模板html (component.html)。

我找到的唯一解决方案是通过“ElementRef”原生元素修改项目。这个解决方案似乎有点复杂,做一些应该很简单的事情。

另一个问题是CSS必须在组件作用域之外定义,这破坏了组件封装。

有没有更简单的解决办法?比如<root [class]="…"" >…模板中的</ root>。


当前回答

对于多职业的情况,就像上面提到的@jbojcic,你可以使用:

主机:{类:'A B C'}

其他回答

下面是我的做法(Angular 7):

在组件中,添加一个输入:

@Input() componentClass: string = '';

然后在组件的HTML模板中添加如下内容:

<div [ngClass]="componentClass">...</div>

最后在你实例化组件的HTML模板中:

<root componentClass="someclass someotherclass">...</root>

免责声明:我是Angular的新手,所以我可能只是运气好而已!

这样你就不需要在组件外添加CSS了:

@Component({
   selector: 'body',
   template: 'app-element',
   // prefer decorators (see below)
   // host:     {'[class.someClass]':'someField'}
})
export class App implements OnInit {
  constructor(private cdRef:ChangeDetectorRef) {}
  
  someField: boolean = false;
  // alternatively also the host parameter in the @Component()` decorator can be used
  @HostBinding('class.someClass') someField: boolean = false;

  ngOnInit() {
    this.someField = true; // set class `someClass` on `<body>`
    //this.cdRef.detectChanges(); 
  }
}

砰砰作响的例子

这个CSS是在组件内部定义的,只有当类someClass在宿主元素上设置时(从外部),选择器才会应用:

:host(.someClass) {
  background-color: red;
}

另一个问题是CSS必须在组件作用域之外定义,这破坏了组件封装

这是不对的。使用scss (SASS),您可以轻松地样式组件(本身;主机)如下:

:host {
    display: block;
    position: absolute;
    width: 100%;
    height: 100%;
    pointer-events: none;
    visibility: hidden;

    &.someClass {
        visibility: visible;
    }
}

这样封装是“完整的”。

对于多职业的情况,就像上面提到的@jbojcic,你可以使用:

主机:{类:'A B C'}

当你有多个主机绑定类时,我发现下面使用HostBinding getter是最方便的:

@Component({ ... })
export class MyComponent {
    @Input()
    theme: 'success' | 'error';

    @HostBinding('class')
    get classes(): Record<string, boolean> {
        'my-component': true,
        'my-component-success-theme': this.theme == 'success',
        'my-component-error-theme': this.theme == 'error' 
    }
}

即使对于没有相应逻辑的普通类绑定,这也是在一个地方管理所有宿主类的好方法。

@Component({ ... })
export class MyComponent {
    @HostBinding('class')
    get classes(): Record<string, boolean> {
        'my-component': true,
        'my-component-modifier': true
    }
}

此外,与@Component装饰器的host属性不同,这种方法还允许您为类字符串使用枚举。

export enum MyComponentClass {
    BaseClass = 'my-component',
    ModifierClass = 'my-component-modifier'
}

@Component({ ... })
export class MyComponent {
    @HostBinding('class')
    get classes(): Record<string, boolean> {
        [MyComponentClass.BaseClass]: true,
        [MyComponentClass.ModifierClass]: true
    }
}