Как использовать [(ngModel)] для содержимого div в angular2?

Я пытаюсь использовать ngModel для двусторонней привязки содержимого входящего содержимого div следующим образом:

<div id="replyiput" class="btn-input"  [(ngModel)]="replyContent"  contenteditable="true" data-text="type..." style="outline: none;"    ></div> 

но он не работает и возникает ошибка:

EXCEPTION: No value accessor for '' in [ddd in PostContent@64:141]
app.bundle.js:33898 ORIGINAL EXCEPTION: No value accessor for ''

person Kim Wong    schedule 13.02.2016    source источник


Ответы (8)


NgModel ожидает, что связанный элемент будет иметь свойство value, которого нет у divs. Вот почему вы получаете ошибку No value accessor.

Вы можете настроить собственное эквивалентное свойство и привязку данных событий, используя свойство textContent (вместо value) и событие input:

import {Component} from 'angular2/core';
@Component({
  selector: 'my-app',
  template: `{{title}}
    <div contenteditable="true" 
     [textContent]="model" (input)="model=$event.target.textContent"></div>
    <p>{{model}}`
})
export class AppComponent {
  title = 'Angular 2 RC.4';
  model = 'some text';
  constructor() { console.clear(); }
}

Plunker

Я не знаю, поддерживается ли событие input во всех браузерах для contenteditable. Вместо этого вы всегда можете привязаться к какому-либо событию клавиатуры.

person Mark Rajcok    schedule 13.02.2016
comment
Спасибо за ваш ответ. Но это не двусторонняя привязка. Когда пользователь вводит что-то во входных данных, переменная модели не изменяется. - person Kim Wong; 14.02.2016
comment
@KimWong, model var определенно меняется в предоставленном мною Plunker. Вот почему я поместил {{model}} в представление / шаблон, чтобы мы могли видеть его изменение при редактировании div. - person Mark Rajcok; 15.02.2016
comment
Спасибо! Это помогает. По какой-то причине angular2rc1 не нравится textContent, вместо этого использование innerText отлично работает. - person Meir; 04.07.2016
comment
Большая помощь, спасибо! Любой способ легко поддерживать многострочность и предотвращать создание текста в начале строки, то есть в обратном направлении? - person Zach Botterman; 12.07.2016
comment
@ZachBotterman, если я добавлю styles: ['div { white-space: pre}' ] к метаданным компонента, это вроде как работает. Мне нужно ввести shift-enter, чтобы новая строка не заменялась. Проверено только в Chrome. - person Mark Rajcok; 12.07.2016
comment
@MarkRajcok Спасибо за ответ. Проверю как можно скорее - person Zach Botterman; 18.07.2016
comment
Я столкнулся с той же проблемой, она работала немного по-другому, отвечая ниже. - person tobek; 21.12.2016
comment
Независимо от события, используемого для запуска model = $ event.target.textContent, в настоящее время это не работает должным образом в Firefox и Edge. Курсор всегда устанавливается на индекс 0 при наборе текста. Вы должны знать об этом. - person Lys; 03.04.2017
comment
Спасибо за ваш ответ. У меня есть более сложный вариант использования. У меня также есть директива уровня атрибута в том же div, и мне нужно иметь двухстороннюю привязку для модели. Так что моя директива получит обновленную модель, и если я изменю модель из директивы, это отразится в пользовательском интерфейсе. Как мне это сделать? - person user911; 17.05.2017
comment
ребята, кто-нибудь знает, как разобраться, чтобы индекс курсора не был постоянно установлен на 0? - person Chris Tarasovs; 04.06.2017
comment
Это не работает в IE. Просто откройте плункер в IE. - person Ziggler; 14.12.2017
comment
в настоящее время это полезно только для набора текста в обратном направлении - person szaman; 05.03.2018
comment
@MarkRajcok Отличное решение. Но ваш плункер не работает нормально в firefox. Он набирает обратный текст в firefox. Я добавил для этого сообщение: stackoverflow.com/questions/51206076/ Нужна ваша помощь. - person Alwaysalearner; 18.07.2018

Обновленный ответ (09.10.2017):

Теперь у меня есть модуль ng-contenteditable. Его совместимость с угловыми формами.

Старый ответ (11.05.2017). В моем случае я могу просто сделать:

<div
  contenteditable="true"
  (input)="post.postTitle = $event.target.innerText"
  >{{ postTitle }}</div>

Где post - это объект со свойством postTitle.

В первый раз, после ngOnInit() и получения post из бэкэнда, я установил this.postTitle = post.postTitle в своем компоненте.

person ktretyak    schedule 11.05.2017
comment
Это не сработает, если вы позволите пользователю редактировать свой ответ после перехода на другой экран (предварительно заданные значения). - person Muhammad bin Yusrat; 07.07.2020

Рабочий Plunkr здесь http://plnkr.co/edit/j9fDFc, но соответствующий код ниже.


Привязка и ручное обновление textContent у меня не работало, оно не обрабатывает разрывы строк (в Chrome при вводе текста после разрыва строки курсор возвращается в начало), но я смог заставить его работать с помощью директивы contenteditable модели из https://www.namekdev.net/2016/01/two-way-binding-to-contenteditable-element-in-angular-2/.

Я настроил его для обработки многострочного простого текста (с \ns, а не <br>s) с помощью white-space: pre-wrap и обновил его, чтобы использовать keyup вместо blur. Обратите внимание, что некоторые решения этой проблемы используют событие input, которое пока не поддерживается в IE или Edge для элементов contenteditable.

Вот код:

Директива:

import {Directive, ElementRef, Input, Output, EventEmitter, SimpleChanges} from 'angular2/core';

@Directive({
  selector: '[contenteditableModel]',
  host: {
    '(keyup)': 'onKeyup()'
  }
})
export class ContenteditableModel {
  @Input('contenteditableModel') model: string;
  @Output('contenteditableModelChange') update = new EventEmitter();

  /**
   * By updating this property on keyup, and checking against it during
   * ngOnChanges, we can rule out change events fired by our own onKeyup.
   * Ideally we would not have to check against the whole string on every
   * change, could possibly store a flag during onKeyup and test against that
   * flag in ngOnChanges, but implementation details of Angular change detection
   * cycle might make this not work in some edge cases?
   */
  private lastViewModel: string;

  constructor(private elRef: ElementRef) {
  }

  ngOnChanges(changes: SimpleChanges) {
    if (changes['model'] && changes['model'].currentValue !== this.lastViewModel) {
      this.lastViewModel = this.model;
      this.refreshView();
    }
  }

  /** This should probably be debounced. */
  onKeyup() {
    var value = this.elRef.nativeElement.innerText;
    this.lastViewModel = value;
    this.update.emit(value);
  }

  private refreshView() {
    this.elRef.nativeElement.innerText = this.model
  }
}

Использование:

import {Component} from 'angular2/core'
import {ContenteditableModel} from './contenteditable-model'

@Component({
  selector: 'my-app',
  providers: [],
  directives: [ContenteditableModel],
  styles: [
    `div {
      white-space: pre-wrap;

      /* just for looks: */
      border: 1px solid coral;
      width: 200px;
      min-height: 100px;
      margin-bottom: 20px;
    }`
  ],
  template: `
    <b>contenteditable:</b>
    <div contenteditable="true" [(contenteditableModel)]="text"></div>

    <b>Output:</b>
    <div>{{text}}</div>

    <b>Input:</b><br>
    <button (click)="text='Success!'">Set model to "Success!"</button>
  `
})
export class App {
  text: string;

  constructor() {
    this.text = "This works\nwith multiple\n\nlines"
  }
}

Пока что тестировалось только в Chrome и FF на Linux.

person tobek    schedule 21.12.2016
comment
Протестировано также в Firefox с Windows, в Ionic 2, и ваш код также работает там. Спасибо! - person Cel; 20.04.2017

Вот другая версия, основанная на ответе @ tobek, который также поддерживает HTML и вставку:

import {
  Directive, ElementRef, Input, Output, EventEmitter, SimpleChanges, OnChanges,
  HostListener, Sanitizer, SecurityContext
} from '@angular/core';

@Directive({
  selector: '[contenteditableModel]'
})
export class ContenteditableDirective implements OnChanges {
  /** Model */
  @Input() contenteditableModel: string;
  @Output() contenteditableModelChange?= new EventEmitter();
  /** Allow (sanitized) html */
  @Input() contenteditableHtml?: boolean = false;

  constructor(
    private elRef: ElementRef,
    private sanitizer: Sanitizer
  ) { }

  ngOnChanges(changes: SimpleChanges) {
    if (changes['contenteditableModel']) {
      // On init: if contenteditableModel is empty, read from DOM in case the element has content
      if (changes['contenteditableModel'].isFirstChange() && !this.contenteditableModel) {
        this.onInput(true);
      }
      this.refreshView();
    }
  }

  @HostListener('input') // input event would be sufficient, but isn't supported by IE
  @HostListener('blur')  // additional fallback
  @HostListener('keyup') onInput(trim = false) {
    let value = this.elRef.nativeElement[this.getProperty()];
    if (trim) {
      value = value.replace(/^[\n\s]+/, '');
      value = value.replace(/[\n\s]+$/, '');
    }
    this.contenteditableModelChange.emit(value);
  }

  @HostListener('paste') onPaste() {
    this.onInput();
    if (!this.contenteditableHtml) {
      // For text-only contenteditable, remove pasted HTML.
      // 1 tick wait is required for DOM update
      setTimeout(() => {
        if (this.elRef.nativeElement.innerHTML !== this.elRef.nativeElement.innerText) {
          this.elRef.nativeElement.innerHTML = this.elRef.nativeElement.innerText;
        }
      });
    }
  }

  private refreshView() {
    const newContent = this.sanitize(this.contenteditableModel);
    // Only refresh if content changed to avoid cursor loss
    // (as ngOnChanges can be triggered an additional time by onInput())
    if (newContent !== this.elRef.nativeElement[this.getProperty()]) {
      this.elRef.nativeElement[this.getProperty()] = newContent;
    }
  }

  private getProperty(): string {
    return this.contenteditableHtml ? 'innerHTML' : 'innerText';
  }

  private sanitize(content: string): string {
    return this.contenteditableHtml ? this.sanitizer.sanitize(SecurityContext.HTML, content) : content;
  }
}
person Rene Hamburger    schedule 28.06.2017
comment
Спасибо, но чтобы избежать ExpressionChangedAfterItHasBeenCheckedError, используйте асинхронный EventEmitter в выводе @Output() contenteditableModelChange?= new EventEmitter(true); ссылка на статью. Может быть, ты сможешь обновить свой код. - person Atiris; 22.09.2017

Я повозился с этими решениями и сейчас буду использовать в своем проекте следующее решение:

<div #topicTitle contenteditable="true" [textContent]="model" (input)="model=topicTitle.innerText"></div>

Я предпочитаю использовать ссылочную переменную шаблона вместо "$ event".

Связанная ссылка: https://angular.io/guide/user-input#get-user-input-from-a-template-reference-variable.

person Flo    schedule 06.11.2017
comment
Я также использовал это решение на редактируемом TD. {{модель}}, предложенная некоторыми другими решениями, вызвала у меня проблемы при наборе текста. Он будет динамически обновлять текст, и я получу некоторую искаженную тарабарщину с этим решением, которого не произошло - person p0enkie; 12.03.2018
comment
Мне нравится это решение, но в firefox оно также печатается в обратном порядке. - person illnr; 12.12.2018

В contenteditable я добился двусторонней привязки с помощью события blur и атрибута innerHTML.

в .html:

<div placeholder="Write your message.."(blur)="getContent($event.target.innerHTML)" contenteditable [innerHTML]="content"></div>

In .ts:

getContent(innerText){
  this.content = innerText;
}
person Sahil Ralkar    schedule 27.05.2020

Вот простое решение, если то, что вы привязываете, является строкой, никаких событий не требуется. Просто поместите текстовое поле в ячейку таблицы и выполните привязку к нему. Затем отформатируйте текстовое поле до прозрачного

HTML:

<tr *ngFor="let x of tableList">
    <td>
        <input type="text" [(ngModel)]="x.value" [ngModelOptions]="{standalone: true}">
    </td>
</tr>
person Isaac    schedule 08.06.2018

Для меня было достаточно использовать javascript без объекта ts. HTML:

 <div
        id="custom-input"
        placeholder="Schreiben..."
</div>

TS:

  • чтобы получить входное значение: document.getElementById("custom-input").innerHTML

  • для установки входного значения: document.getElementById("custom-input").innerHTML = "myValue"

И все работает отлично. Мне пришлось использовать div вместо ionic ion-textarea, потому что у меня были проблемы с автоматическим изменением размера. С ion-textarea я смог сделать авторазмер только с помощью js. Теперь я делаю автоматическое изменение размера с помощью CSS, что, на мой взгляд, лучше.

person FAndrew    schedule 20.01.2020