Angular 2 Checkbox Zwei-Wege-Datenbindung

202

Ich bin ziemlich neu in Angular2 und habe ein kleines Problem:

In meinem Login-Component-HTML habe ich zwei Kontrollkästchen, die ich auf zwei Arten an das Login-Component-TypeScript binden möchte.

Dies ist der HTML:

<div class="checkbox">
<label>
    <input #saveUsername [(ngModel)]="saveUsername.selected" type="checkbox" data-toggle="toggle">Save username
</label>
</div>

Und das ist die Component.ts:

import { Component, OnInit }    from '@angular/core';
import { Router }               from '@angular/router';
import { Variables }            from '../../services/variables';

@Component({
    selector: 'login',
    moduleId: module.id,
    templateUrl: 'login.component.html',
    styleUrls: ['login.component.css']
})


export class LoginComponent implements OnInit {

    private saveUsername: boolean = true;
    private autoLogin: boolean = true;
    constructor(private router: Router, private variables: Variables) { }

    ngOnInit() { 
        this.loginValid = false;
        // Get user name from local storage if you want to save

        if (window.localStorage.getItem("username") === null) {
           this.saveUsername = true;
           this.autoLogin = true;
           console.log(this.saveUsername, this.autoLogin);
        } else {
           console.log("init", window.localStorage.getItem("username"));
        }
    }

    login(username: string, password: string, saveUsername: boolean, autoLogin: boolean) {
        this.variables.setUsername(username);
        this.variables.setPassword(password);
        this.variables.setIsLoggedIn(true);
        console.log(saveUsername, autoLogin);
        //this.router.navigate(['dashboard']);
    }

Wenn ich auf ein Kontrollkästchen klicke, erhalte ich den richtigen Wert in der Steuerung (Komponente).

Wenn ich jedoch den Wert von beispielsweise saveUsernamein der Komponente ändere , hat das Kontrollkästchen den neuen Wert nicht "erhalten".

Daher kann ich das Kontrollkästchen in der Komponente nicht bearbeiten (wie ich es ngOnInitin der Komponente tun möchte .

Danke für Ihre Hilfe!

Junias
quelle
Überprüfen Sie hier für Angular 7 Beispiel der Checkbox Bindung freakyjolly.com/…
Code Spy

Antworten:

350

Sie können entfernen .selectedaus saveUsernameIhrer Checkbox Eingabe seit saveUsername ein boolean ist. Anstelle von [(ngModel)]Gebrauch[checked]="saveUsername" (change)="saveUsername = !saveUsername"

Bearbeiten: Richtige Lösung:

<input
  type="checkbox"
  [checked]="saveUsername"
  (change)="saveUsername = !saveUsername"/>

Update: Wie @newman bemerkt hat, wenn ngModeles in einem Formular verwendet wird, funktioniert es nicht. Sie sollten jedoch ein [ngModelOptions]Attribut wie (getestet in Angular 7) verwenden:

<input
  type="checkbox"
  [(ngModel)]="saveUsername"
  [ngModelOptions]="{standalone: true}"/> `

Ich habe auch ein Beispiel bei Stackblitz erstellt: https://stackblitz.com/edit/angular-abelrm

Hakany
quelle
Wenn ich das mache, bekomme ich eine Fehlermeldung: Fehler: Kann keiner Referenz oder Variablen zuweisen!
Junias
hm, ich erhalte bereits folgende Fehlermeldung: Nicht behandelte Ablehnung eines Versprechens: Kann keiner Referenz oder Variablen zugewiesen werden! ;; Zone: <wurzel>; Aufgabe: Promise.then; Wert: Fehler: Kann keiner Referenz oder Variablen zugewiesen werden! Und woher weiß das Kontrollkästchen, ob es ein- oder ausgeschaltet werden soll? In Angular 1 könnte ich es so machen: $scope.loginData = {}; $scope.loginData.username = window.localStorage.getItem("username");und das in der Vorlage:<ion-toggle ng-model="saveUsername" ng-change="toggleSaveUsername()" toggle-class="toggle-energized">Benutzername speichern</ion-toggle>
Junias
1
Aus bestimmten Gründen funktioniert die Eingabe nicht, wenn sie sich nicht in einem Formular befindet, sondern Teil der * ngFor-Schleife ist [(ngModel)]="saveUsername", aber diese funktioniert. Es muss ein Fehler im Winkel sein?
Newman
1
Dies funktioniert bei mir, während [(ngModel)] sich komisch verhält. Kann mich jemand auf eine Dokumentation oder eine Diskussion darüber verweisen, warum [aktiviert] bei Kontrollkästchen besser zu verwenden ist als ngModel?
vc669
3
Wenn ngModeles in einem Formular verwendet wird, funktioniert es nicht. [ngModelOptions]="{standalone: true}"ist was ich brauchte.
rainversion_3
88

Leider ist die von @hakani bereitgestellte Lösung keine bidirektionale Bindung . Es wird nur das Einwegmodell aus dem UI / FrontEnd-Teil geändert.

Stattdessen das einfache:

<input [(ngModel)]="checkboxFlag" type="checkbox"/>

führt eine bidirektionale Bindung für das Kontrollkästchen durch.

Wenn anschließend das Modell checkboxFlag von Backend oder UI-Teil voila geändert wird, speichert checkboxFlag den tatsächlichen Status des Kontrollkästchens.

Um sicherzugehen, dass ich Plunker-Code vorbereitet habe, um das Ergebnis zu präsentieren: https://plnkr.co/edit/OdEAPWRoqaj0T6Yp0Mfk

Um diese Antwort zu vervollständigen, sollten Sie das Array " import { FormsModule } from '@angular/forms'In" app.module.tsund "Import" hinzufügen

import { FormsModule } from '@angular/forms';

[...]

@NgModule({
  imports: [
    [...]
    FormsModule
  ],
  [...]
})
Ruslan Makrenko
quelle
3
Scheint nicht zu funktionieren, wenn Sie das Kontrollkästchen in einem verwenden ngFor, während Sie eine Reihe von Objekten wie[{"checked":true},{"checked":false}]
jackOfAll
Ich kann diese Lösung nicht zum cUncaught Error: Template parse errors: Can't bind to 'ngModel' since it isn't a known property of 'input'
Laufen bringen
5
@sebnukem Es scheint, dass Sie den Import für FormsModule deklarieren.
Teddy
@sebnukem innerhalb von <input> ist eine Namenseigenschaft erforderlich, damit [(ngModel)] verwendet werden kann
Angela Pan
Funktioniert es auch für mehrere Kontrollkästchen? <input type="checkbox" [(ngModel)]="day.IsChecked" [checked]="day.IsChecked" />
Barnes
35

Ich arbeite mit Angular5 und musste das Attribut "name" hinzufügen, damit die Bindung funktioniert ... Die "id" ist für die Bindung nicht erforderlich.

<input type="checkbox" id="rememberMe" name="rememberMe" [(ngModel)]="rememberMe">
Kris Kilton
quelle
1
Diese Antwort hat bei mir funktioniert. Andere Antworten sind ebenfalls richtig. Aber mit [(ngModel)] müssen wir keine separate Funktion schreiben, um den Booleschen
Wert
25

Ich bevorzuge etwas expliziteres:

component.html

<input #saveUserNameCheckBox
    id="saveUserNameCheckBox" 
    type="checkbox" 
    [checked]="saveUsername" 
    (change)="onSaveUsernameChanged(saveUserNameCheckBox.checked)" />

component.ts

public saveUsername:boolean;

public onSaveUsernameChanged(value:boolean){
    this.saveUsername = value;
}
MovGP0
quelle
2
Das einzige, was in diesem ganzen Thread funktioniert hat. Danke dir! Angular verwenden 8.2.11.
Daka
Diese Antwort sollte oben sein.
Hardit Raghav
1
rettete meinen Tag :-)
hannes neukermans
6

Bei Verwendung der <abc [(bar)]="foo"/>Syntax für Winkel.

Dies bedeutet: <abc [bar]="foo" (barChange)="foo = $event" />

Was bedeutet, dass Ihre Komponente haben sollte:

@Input() bar;
@Output() barChange = new EventEmitter();
Jose Alberto Fernandez
quelle
Antwort ist wirklich gut - kurz und klar, sollte aber unter der anderen Frage
Nikita
5

Sie können einfach so etwas verwenden, um eine bidirektionale Datenbindung zu erhalten:

<input type="checkbox" [checked]="model.property" (change)="model.property = !model.consent_obtained_ind">
Nuno Ferro
quelle
3

Um das Kontrollkästchen zum Laufen zu bringen, sollten Sie alle folgenden Schritte ausführen:

  1. FormsModule in Ihr Modul importieren
  2. Fügen Sie die Eingabe in ein formTag ein
  3. Ihre Eingabe sollte folgendermaßen aussehen:

    <input name="mpf" type="checkbox" [(ngModel)]="value" />

    Hinweis: Vergessen Sie nicht, den Namen in Ihre Eingabe einzugeben.

Mahdi Shahbazi
quelle
3

Sie müssen name="selected"dem inputElement ein Attribut hinzufügen .

Beispielsweise:

<div class="checkbox">
  <label>
    <input name="selected" [(ngModel)]="saveUsername.selected" type="checkbox">Save username
  </label>
</div>
M. Shakeri
quelle
2

Ich habe eine benutzerdefinierte Komponente versucht, die bidirektionale Bindung

Meine Komponente: <input type="checkbox" [(ngModel)]="model" >

_model:  boolean;   

@Output() checked: EventEmitter<boolean> = new EventEmitter<boolean>();

@Input('checked')
set model(checked: boolean) {

  this._model = checked;
  this.checked.emit(this._model);
  console.log('@Input(setmodel'+checked);
}

get model() {
  return this._model;
}

Seltsame Sache ist, dass dies funktioniert

<mycheckbox  [checked]="isChecked" (checked)="isChecked = $event">

während dies nicht wird

<mycheckbox  [(checked)]="isChecked">
user1786641
quelle
1

Wenn Sie in jedem Fall einen Wert mit einem Kontrollkästchen binden müssen, das nicht boolesch ist, können Sie die folgenden Optionen ausprobieren

In der HTML-Datei:

<div class="checkbox">
<label for="favorite-animal">Without boolean Value</label>
<input type="checkbox" value="" [checked]="ischeckedWithOutBoolean == 'Y'" 
(change)="ischeckedWithOutBoolean = $event.target.checked ? 'Y': 'N'">
</div>

in der KomponenteischeckedWithOutBoolean: any = 'Y';

Siehe im stackblitz https://stackblitz.com/edit/angular-5szclb?embed=1&file=src/app/app.component.html

Sabbir
quelle
1

Ich weiß, dass es eine wiederholte Antwort sein kann, aber für jeden, der eine Liste von Kontrollkästchen mit einem Auswahlkästchen in Winkelform laden möchte, folge ich diesem Beispiel: Aktivieren Sie das Kontrollkästchen Alle / deaktivieren Sie alle mit Winkel 2+

es funktioniert gut, muss aber nur hinzugefügt werden

[ngModelOptions]="{standalone: true}" 

Der endgültige HTML-Code sollte folgendermaßen aussehen:

<ul>
    <li><input type="checkbox" [(ngModel)]="selectedAll" (change)="selectAll();"/></li>
    <li *ngFor="let n of names">
    <input type="checkbox" [(ngModel)]="n.selected" (change)="checkIfAllSelected();">{{n.name}}
    </li>
  </ul>

Typoskript

  selectAll() {
    for (var i = 0; i < this.names.length; i++) {
      this.names[i].selected = this.selectedAll;
    }
  }
  checkIfAllSelected() {
    this.selectedAll = this.names.every(function(item:any) {
        return item.selected == true;
      })
  }

hoffe das hilft thnx

Taha Mosaad
quelle
0

Eine Problemumgehung, um dasselbe zu erreichen, insbesondere wenn Sie das Kontrollkästchen mit der for-Schleife verwenden möchten, besteht darin, den Status des Kontrollkästchens in einem Array zu speichern und es basierend auf dem Index der *ngForSchleife zu ändern . Auf diese Weise können Sie den Status des Kontrollkästchens in Ihrer Komponente ändern.

app.component.html

<div *ngFor="let item of items; index as i"> <input type="checkbox" [checked]="category[i]" (change)="checkChange(i)"> {{item.name}} </div>

app.component.ts

items = [
    {'name':'salad'},
    {'name':'juice'},
    {'name':'dessert'},
    {'name':'combo'}
  ];

  category= []

  checkChange(i){
    if (this.category[i]){  
      this.category[i] = !this.category[i];
    }
    else{
      this.category[i] = true;
    }
  }
Sid
quelle
0

Meine Winkelanweisung wie anglejs (ng-true-value ng-false-value)

@Directive({
    selector: 'input[type=checkbox][checkModel]'
})
export class checkboxDirective {
    @Input() checkModel:any;
    @Input() trueValue:any;
    @Input() falseValue:any;
    @Output() checkModelChange = new EventEmitter<any>();

    constructor(private el: ElementRef) { }

    ngOnInit() {
       this.el.nativeElement.checked = this.checkModel==this.trueValue;
    }

    @HostListener('change', ['$event']) onChange(event:any) {
        this.checkModel = event.target.checked ? this.trueValue : this.falseValue;
        this.checkModelChange.emit(this.checkModel);
    }

}

html

<input type="checkbox" [(checkModel)]="check" [trueValue]="1" [falseValue]="0">
Edgardo Rivera
quelle
Hallo, bitte erläutern Sie Ihr Problem, was haben Sie versucht und fügen Sie Ihrer Frage einen Kontext hinzu
538ROMEO
0

Im Winkel-p-Kontrollkästchen

Verwenden Sie alle Attribute des Kontrollkästchens p

<p-checkbox name="checkbox" value="isAC" 
    label="All Colors" [(ngModel)]="selectedAllColors" 
    [ngModelOptions]="{standalone: true}" id="al" 
    binary="true">
</p-checkbox>

Und was noch wichtiger ist, vergessen Sie nicht, [ngModelOptions]="{standalone: true}so gut wie es meinen Tag gerettet hat.

Mian Taimoor Tahir
quelle
0
Angular: "9.0.0"
Angular CLI: 9.0.1
Node: 13.10.1
OS: linux x64

.html Datei

<input [(ngModel)]="userConsent" id="userConsent" required type="checkbox"/> " I Accept"

.ts Datei

userConsent: boolean = false;
Khizer
quelle