Angular: classe condicional com * ngClass

562

O que há de errado com meu código angular? Estou obtendo:

Cannot read property 'remove' of undefined at BrowserDomAdapter.removeClass ...

HTML

<ol class="breadcrumb">
    <li *ngClass="{active: step==='step1'}" (click)="step='step1; '">Step1</li>
    <li *ngClass="{active: step==='step2'}"  (click)="step='step2'">Step2</li>
    <li *ngClass="{active: step==='step3'}" (click)="step='step3'">Step3</li>
</ol>
daniel
fonte

Respostas:

1229

A versão angular 2, ..., 9 fornece várias maneiras de adicionar classes condicionalmente:

digite um

[class.my-class]="step === 'step1'"

tipo dois

[ngClass]="{'my-class': step === 'step1'}"

e opção múltipla:

[ngClass]="{'my-class': step === 'step1', 'my-class2':step === 'step2' }"

tipo três

[ngClass]="{1:'my-class1',2:'my-class2',3:'my-class4'}[step]"

tipo quatro

[ngClass]="(step=='step1')?'my-class1':'my-class2'"
MostafaMashayekhi
fonte
7
Resposta perfeita, basta corrigir o tipo 2 para: [ngClass] = "{'my-class': step == 'step1'}" Com o '' int o nome da classe
Adriano Galesso Alves
2
Para o tipo três, a ordem do nome da classe e a verificação estão incorretas. Deve ser o nome da classe primeiro, como [ngClass] = "{'my-class1': 1, 'my-class2': 2}"
obaylis 18/07/2012
1
olhares como "tipo três" e "digite quatro" são usos específicos de [ngClass]="js expression returning html class string"modo que são os mesmos neste sentido
YakovL
1
Obrigado cara .. você é demais
Pranav MS
1
Resposta perfeita, companheiro. Muito obrigado !
Anjana Silva 15/01
423

[ngClass]=...em vez de *ngClass.

* é apenas para a sintaxe abreviada de diretivas estruturais em que você pode, por exemplo, usar

<div *ngFor="let item of items">{{item}}</div>

em vez da versão equivalente mais longa

<template ngFor let-item [ngForOf]="items">
  <div>{{item}}</div>
</template>

Consulte também https://angular.io/docs/ts/latest/api/common/index/NgClass-directive.html

<some-element [ngClass]="'first second'">...</some-element>
<some-element [ngClass]="['first', 'second']">...</some-element>
<some-element [ngClass]="{'first': true, 'second': true, 'third': false}">...</some-element>
<some-element [ngClass]="stringExp|arrayExp|objExp">...</some-element>
<some-element [ngClass]="{'class1 class2 class3' : true}">...</some-element>

Veja também https://angular.io/docs/ts/latest/guide/template-syntax.html

<!-- toggle the "special" class on/off with a property -->
<div [class.special]="isSpecial">The class binding is special</div>

<!-- binding to `class.special` trumps the class attribute -->
<div class="special"
     [class.special]="!isSpecial">This one is not so special</div>
<!-- reset/override all class names with a binding  -->
<div class="bad curly special"
     [class]="badCurly">Bad curly</div>
Günter Zöchbauer
fonte
1
A partir da documentação angular: "O asterisco é" açúcar sintático "para algo um pouco mais complicado. Internamente, o Angular converte o atributo * ngIf em um elemento <ng-template> envolvido no elemento host, como este. A diretiva * ngIf foi movido para o elemento <ng-template>, onde se tornou uma ligação de propriedade, [ngIf]. O restante do <div>, incluindo seu atributo de classe, foi movido para dentro do elemento <ng-template>. " - more info @ angular.io/guide/structural-directives#the-asterisk--prefix
Combine
Na verdade, não é nada mais complicado, *apenas permite uma sinax simplificada em vez de uma forma canônica.
Günter Zöchbauer
75

Outra solução estaria usando [class.active].

Exemplo:

<ol class="breadcrumb">
    <li [class.active]="step=='step1'" (click)="step='step1'">Step1</li>
</ol>
Joel Almeida
fonte
8
Eu acho que essa deve ser a resposta aceita, pois é a maneira Angular2 de definir a classe html (que eu não conhecia e o google me trouxe aqui).
kub1x
62

Essa é a estrutura normal para ngClassé:

[ngClass]="{'classname' : condition}"

Portanto, no seu caso, use-o assim ...

<ol class="breadcrumb">
  <li [ngClass]="{'active': step==='step1'}" (click)="step='step1'">Step1</li>
  <li [ngClass]="{'active': step==='step2'}" (click)="step='step2'">Step2</li>
  <li [ngClass]="{'active': step==='step3'}" (click)="step='step3'">Step3</li>
</ol>
Alireza
fonte
48

com os exemplos a seguir, você pode usar 'IF ELSE'

<p class="{{condition ? 'checkedClass' : 'uncheckedClass'}}">
<p [ngClass]="condition ? 'checkedClass' : 'uncheckedClass'">
<p [ngClass]="[condition ? 'checkedClass' : 'uncheckedClass']">
Chaitanya Nekkalapudi
fonte
1
Eu tentei a primeira e a segunda solução. Apenas o segundo funcionou para mim #
22127878
36

Você pode usar o ngClass para aplicar o nome da classe condicionalmente e não no Angular

Por exemplo

[ngClass]="'someClass'">

Condicional

[ngClass]="{'someClass': property1.isValid}">

Condição múltipla

 [ngClass]="{'someClass': property1.isValid && property2.isValid}">

Expressão de método

[ngClass]="getSomeClass()"

Este método estará dentro do seu componente

 getSomeClass(){
        const isValid=this.property1 && this.property2;
        return {someClass1:isValid , someClass2:isValid};
    }
Code-EZ
fonte
14

Você deve usar algo (em [ngClass]vez de *ngClass) assim:

<ol class="breadcrumb">
  <li [ngClass]="{active: step==='step1'}" (click)="step='step1; '">Step1</li>
  (...)

Thierry Templier
fonte
10

No Angular 7.X

As classes CSS são atualizadas da seguinte forma, dependendo do tipo de avaliação da expressão:

  • string - as classes CSS listadas na string (delimitadas por espaço) são adicionadas

  • Matriz - as classes CSS declaradas como elementos da matriz são adicionadas

  • As chaves de objeto são classes CSS que são adicionadas quando a expressão fornecida no valor é avaliada como um valor verdadeiro, caso contrário, elas são removidas.

<some-element [ngClass]="'first second'">...</some-element>

<some-element [ngClass]="['first', 'second']">...</some-element>

<some-element [ngClass]="{'first': true, 'second': true, 'third': false}">...</some-element>

<some-element [ngClass]="stringExp|arrayExp|objExp">...</some-element>

<some-element [ngClass]="{'class1 class2 class3' : true}">...</some-element>
Rohit.007
fonte
7

para estender MostafaMashayekhi sua resposta para a opção dois> você também pode encadear várias opções com um ','

[ngClass]="{'my-class': step=='step1', 'my-class2':step=='step2' }"

Também * ngIf pode ser usado em algumas dessas situações geralmente combinadas com um * ngFor

class="mats p" *ngIf="mat=='painted'"
Robert Leeuwerink
fonte
6

Enquanto criava um formulário reativo, tive que atribuir 2 tipos de classe no botão. Foi assim que eu fiz:

<button type="submit" class="btn" [ngClass]="(formGroup.valid)?'btn-info':''" 
[disabled]="!formGroup.valid">Sign in</button>

Quando o formulário é válido, o botão possui as classes btn e btn (do bootstrap), caso contrário, apenas a classe btn.

Sarvar Nishonboev
fonte
5

Além disso, você pode adicionar com a função de método:

Em HTML

<div [ngClass]="setClasses()">...</div>

Em component.ts

// Set Dynamic Classes
  setClasses() {
    let classes = {
      constantClass: true,
      'conditional-class': this.item.id === 1
    }

    return classes;
  }
Alper BULUT
fonte
4

Deixe, YourCondition é sua condição ou uma propriedade booleana, faça assim

[class.yourClass]="YourCondition"
Abdus Salam Azad
fonte
4

ngClass sintaxe:

[ngClass]="{'classname' : conditionFlag}"

Você pode usar assim:

<ol class="breadcrumb">
  <li [ngClass]="{'active': step==='step1'}" (click)="step='step1'">Step1</li>
  <li [ngClass]="{'active': step==='step2'}" (click)="step='step2'">Step2</li>
  <li [ngClass]="{'active': step==='step3'}" (click)="step='step3'">Step3</li>
</ol>
Chirag
fonte
4

Isto é o que funcionou para mim:

[ngClass]="{'active': dashboardComponent.selected_menu == 'profile'}"
Ninad Kulkarni
fonte
4

Você pode usar [ngClass] ou [class.classname], ambos funcionarão da mesma maneira.
[class.my-class]="step==='step1'"

   OU

[ngClass]="{'my-class': step=='step1'}"

Ambos funcionarão da mesma maneira!

Waleed Shahzaib
fonte
1

Não é relevante com a [ngClass]diretiva, mas eu também estava recebendo o mesmo erro que

Não é possível ler a propriedade 'remover' de indefinido em ...

e eu pensei que era o erro na minha [ngClass]condição, mas a propriedade que eu estava tentando acessar na condição de [ngClass]não foi inicializada.

Como se eu tivesse isso no meu arquivo datilografado

element: {type: string};

e no meu [ngClass]eu estava usando

[ngClass]="{'active', element.type === 'active'}"

e eu estava recebendo o erro

Não é possível ler a propriedade 'type' de undefined em ...

e a solução foi consertar minha propriedade

element: {type: string} = {type: 'active'};

Espero que ajude alguém que está tentando corresponder a uma condição de uma propriedade em [ngClass]

Hamza Khanzada
fonte
1

<div class="collapse in " [ngClass]="(active_tab=='assignservice' || active_tab=='manage')?'show':''" id="collapseExampleOrganization" aria-expanded="true" style="">
 <ul> 	 <li class="nav-item" [ngClass]="{'active': active_tab=='manage'}">
<a routerLink="/main/organization/manage" (click)="activemenu('manage')"> <i class="la la-building-o"></i>
<p>Manage</p></a></li> 
<li class="nav-item" [ngClass]="{'active': active_tab=='assignservice'}"><a routerLink="/main/organization/assignservice" (click)="activemenu('assignservice')"><i class="la la-user"></i><p>Add organization</p></a></li>
</ul></div>

O código é um bom exemplo de ngClass se houver outra condição.

[ngClass]="(active_tab=='assignservice' || active_tab=='manage')?'show':''"

[ngClass]="{'active': active_tab=='assignservice'}"
amarbhanu
fonte
0

Tente assim ..

Defina sua turma com ''

<ol class="breadcrumb">
    <li *ngClass="{'active': step==='step1'}" (click)="step='step1; '">Step1</li>
    <li *ngClass="{'active': step==='step2'}"  (click)="step='step2'">Step2</li>
    <li *ngClass="{'active': step==='step3'}" (click)="step='step3'">Step3</li>
</ol>
Aishwarya Kathavarayan
fonte