* ngIf с асинхронным каналом для наблюдаемого из BehaviorSubject, по-видимому, неправильно активируется до загрузки данных - PullRequest
0 голосов
/ 21 апреля 2019

Если массив в ответе пуст, я хочу отобразить сообщение, иначе элементы массива.

В случае, если ответ имеет непустой массив, условие * ngIf переходит в true и сообщение отображается до тех пор, пока данные не будут загружены.

Фрагмент шаблона HTML:

<div *ngIf="personalAndStarredBookmarks$ && (personalAndStarredBookmarks$ | async)?.size === 0; else personalBookmarksList" class="missing-category-bookmarks-message  alert alert-info">
  <p>No bookmarks yet</p>
</div>
<ng-template #personalBookmarksList>
  <app-async-bookmark-list [bookmarks]="personalAndStarredBookmarks$" [shownSize]="10" [userData]="userData"></app-async-bookmark-list>
</ng-template>

Фрагмент компонента:

import { List } from 'immutable';

export class PersonalBookmarksListComponent implements OnInit {

  personalAndStarredBookmarks$: Observable<List<Bookmark>>;

  ngOnInit(): void {

    this.personalAndStarredBookmarks$ = this.personalBookmarksStore.getPersonalBookmarks();
  }
  ...
}

Хранилище использует BehaviourSubject , где хранится ответ службы, которая фактически выполняет HTTP-вызов.

Фрагмент магазина

@Injectable()
export class PersonalBookmarksStore {

  private _personalBookmarks: BehaviorSubject<List<Bookmark>> = new BehaviorSubject(List([]));

  constructor(private personalBookmarkService: PersonalBookmarkService,
              private keycloakService: KeycloakService
  ) {
    keycloakService.loadUserProfile().then(keycloakProfile => {
      this.userId = keycloakProfile.id;
      this.loadInitialData();
    });
  }

  private loadInitialData() {
    this.personalBookmarkService.getAllPersonalBookmarks(this.userId)
      .subscribe(
        data => {
          let bookmarks: Bookmark[] = <Bookmark[]>data;
          this._personalBookmarks.next(List(bookmarks));
        },
        err => console.error('Error retrieving bookmarks', err)
      );
  }

  getPersonalBookmarks(): Observable<List<Bookmark>> {
    return this._personalBookmarks.asObservable();
  }
  ...
}

Если я вызываю Службу напрямую (а не в Хранилище) в компоненте, она ведет себя как ожидалось ...

Ответы [ 2 ]

0 голосов
/ 22 апреля 2019

Виновником является инициализация объекта BehaviorSubject с пустым списком, который отправляется и условие оценивается как true:

private _personalBookmarks: BehaviorSubject<List<Bookmark>> = new BehaviorSubject(List([]));

Исправлено - инициация объекта поведения с null или undefined

private _personalBookmarks: BehaviorSubject<List<Bookmark>> = new BehaviorSubject(null);
0 голосов
/ 22 апреля 2019

Вы должны реструктурировать свой шаблон для эффективной обработки асинхронных вызовов, назначив локальную переменную как

<ng-content*ngIf="personalAndStarredBookmarks$ | async as personalAndStarredBookmarks">
    <div *ngIf="personalAndStarredBookmarks.size === 0; else personalBookmarksList" 
        class="missing-category-bookmarks-message  alert alert-info">
        <p>No bookmarks yet</p>
    </div>
    <ng-template #personalBookmarksList>
        <app-async-bookmark-list [bookmarks]="personalAndStarredBookmarks" [shownSize]="10"
            [userData]="userData">
        </app-async-bookmark-list>
    </ng-template>
<ng-content>

При этом вы сможете эффективно обрабатывать все условия и загружать основное содержимое только после того, как асинхронный вызовразрешено.

См. ДЕМО ЗДЕСЬ

...