Angular: clase condicional con * ngClass

562

¿Qué tiene de malo mi código angular? Estoy obteniendo:

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
fuente

Respuestas:

1230

La versión angular 2, ..., 9 proporciona varias formas de agregar clases condicionalmente:

tipo uno

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

tipo dos

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

y opción múltiple:

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

tipo tres

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

tipo cuatro

[ngClass]="(step=='step1')?'my-class1':'my-class2'"
MostafaMashayekhi
fuente
77
Respuesta perfecta, solo arregle el tipo 2 a: [ngClass] = "{'my-class': step == 'step1'}" Con el '' int el nombre de la clase
Adriano Galesso Alves
2
Para el tipo tres, el orden del nombre de la clase y la verificación es incorrecto. Primero debe ser el nombre de la clase, como [ngClass] = "{'my-class1': 1, 'my-class2': 2}"
obaylis
1
parece que "tipo tres" y "tipo cuatro" son usos específicos de [ngClass]="js expression returning html class string"lo que son lo mismo en este sentido
YakovL
1
Gracias hombre ... eres increíble
Pranav MS
1
Respuesta perfecta amigo. Muchas gracias !
Anjana Silva
423

[ngClass]=...en lugar de *ngClass.

* es solo para la sintaxis abreviada de directivas estructurales donde puede, por ejemplo, usar

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

en lugar de la versión equivalente más larga

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

Ver también 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>

Ver también 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
fuente
1
De la documentación angular: "El asterisco es" azúcar sintáctico "para algo un poco más complicado. Internamente, Angular traduce el atributo * ngIf en un elemento <ng-template>, envuelto alrededor del elemento host, así. La directiva * ngIf se movió al elemento <ng-template> donde se convirtió en un enlace de propiedad, [ngIf]. El resto de <div>, incluido su atributo de clase, se movió dentro del elemento <ng-template> ". - más información @ angular.io/guide/structural-directives#the-asterisk--prefix
Combine
En realidad, no es nada más complicado, *solo permite una sinax simplificada en lugar de una forma canónica.
Günter Zöchbauer
76

Otra solución sería usar [class.active].

Ejemplo:

<ol class="breadcrumb">
    <li [class.active]="step=='step1'" (click)="step='step1'">Step1</li>
</ol>
Joel Almeida
fuente
8
Creo que esta debería ser la respuesta aceptada, ya que esta es la forma Angular2 de establecer la clase html (que no conocía y google me trajo aquí).
kub1x
62

Esa es la estructura normal para ngClasses:

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

Entonces, en su caso, simplemente utilícelo así ...

<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
fuente
48

con los siguientes ejemplos puede usar 'IF ELSE'

<p class="{{condition ? 'checkedClass' : 'uncheckedClass'}}">
<p [ngClass]="condition ? 'checkedClass' : 'uncheckedClass'">
<p [ngClass]="[condition ? 'checkedClass' : 'uncheckedClass']">
Chaitanya Nekkalapudi
fuente
1
Intenté la primera y la segunda solución. Solo el segundo funcionó para mí
usuario1238784
36

Puede usar ngClass para aplicar el nombre de la clase condicionalmente y no en Angular

Por ejemplo

[ngClass]="'someClass'">

Condicional

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

Condición Múltiple

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

Expresión del método

[ngClass]="getSomeClass()"

Este método estará dentro de su componente

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

Deberías usar algo (en [ngClass]lugar de *ngClass) así:

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

Thierry Templier
fuente
10

En Angular 7.X

Las clases CSS se actualizan de la siguiente manera, según el tipo de evaluación de expresión:

  • cadena: se añaden las clases CSS enumeradas en la cadena (espacio delimitado)

  • Matriz: se añaden las clases CSS declaradas como elementos de matriz

  • Las claves de objeto son clases CSS que se agregan cuando la expresión dada en el valor se evalúa como un valor verdadero, de lo contrario se eliminan.

<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
fuente
7

para extender MostafaMashayekhi su respuesta para la opción dos> también puede encadenar múltiples opciones con un ','

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

También * ngIf se puede usar en algunas de estas situaciones, generalmente combinado con un * ngFor

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

Mientras creaba una forma reactiva, tuve que asignar 2 tipos de clase en el botón. Así es como lo hice:

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

Cuando el formulario es válido, el botón tiene btn y btn-class (de bootstrap), de lo contrario solo btn class.

Sarvar Nishonboev
fuente
5

Además, puede agregar con la función de método:

En HTML

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

En component.ts

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

    return classes;
  }
Alper BULUT
fuente
4

Deje que YourCondition sea su condición o una propiedad booleana, luego haga esto

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

ngClass sintaxis:

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

Puedes usar así:

<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
fuente
4

Esto es lo que funcionó para mí:

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

Puede usar [ngClass] o [class.classname], ambos funcionarán igual.
[class.my-class]="step==='step1'"

   O

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

¡Ambos funcionarán igual!

Waleed Shahzaib
fuente
1

No es relevante con la [ngClass]directiva, pero también recibí el mismo error que

No se puede leer la propiedad 'eliminar' de indefinido en ...

y pensé que era el error en mi [ngClass]condición, pero resultó que la propiedad a la que intentaba acceder [ngClass]no estaba inicializada.

Como si tuviera esto en mi archivo mecanografiado

element: {type: string};

y en mi [ngClass]estaba usando

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

y estaba recibiendo el error

No se puede leer la propiedad 'tipo' de indefinido en ...

y la solución fue arreglar mi propiedad a

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

Espero que ayude a alguien que está tratando de igualar una condición de una propiedad en [ngClass]

Hamza Khanzada
fuente
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>

El código es un buen ejemplo de ngClass si otra condición.

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

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

Intenta así ...

Define tu clase con ''

<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
fuente