Skip to content

Commit

Permalink
fix(option): emit selection change when selection is changed (#1334)
Browse files Browse the repository at this point in the history
  • Loading branch information
yggg authored Apr 8, 2019
1 parent a7a158d commit 788a6d4
Show file tree
Hide file tree
Showing 3 changed files with 237 additions and 51 deletions.
21 changes: 15 additions & 6 deletions src/framework/theme/components/select/option.component.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import {
OnDestroy,
Output,
} from '@angular/core';
import { Observable, Subject } from 'rxjs';
import { convertToBoolProperty } from '../helpers';
import { NbSelectComponent } from './select.component';

Expand Down Expand Up @@ -52,10 +53,18 @@ export class NbOptionComponent<T> implements OnDestroy {
}

/**
* Fires value on click.
* Fires value when option selection change.
* */
@Output() selectionChange: EventEmitter<NbOptionComponent<T>> = new EventEmitter();

/**
* Fires when option clicked
*/
private click$: Subject<NbOptionComponent<T>> = new Subject<NbOptionComponent<T>>();
get click(): Observable<NbOptionComponent<T>> {
return this.click$.asObservable();
}

selected: boolean = false;
disabled: boolean = false;
private alive: boolean = true;
Expand Down Expand Up @@ -96,7 +105,7 @@ export class NbOptionComponent<T> implements OnDestroy {

@HostListener('click')
onClick() {
this.selectionChange.emit(this);
this.click$.next(this);
}

select() {
Expand All @@ -107,18 +116,18 @@ export class NbOptionComponent<T> implements OnDestroy {
this.setSelection(false);
}

private setSelection(isSelected: boolean): void {
private setSelection(selected: boolean): void {
/**
* In case of changing options in runtime the reference to the selected option will be kept in select component.
* This may lead to exceptions with detecting changes in destroyed component.
*
* Also Angular can call writeValue on destroyed view (select implements ControlValueAccessor).
* angular/angular#27803
* */
if (this.alive) {
this.selected = isSelected;
if (this.alive && this.selected !== selected) {
this.selected = selected;
this.selectionChange.emit(this);
this.cd.markForCheck();
}
}
}

53 changes: 28 additions & 25 deletions src/framework/theme/components/select/select.component.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ import {
} from '@angular/core';
import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';
import { merge, Observable, Subject } from 'rxjs';
import { takeWhile } from 'rxjs/operators';
import { startWith, switchMap, takeWhile } from 'rxjs/operators';

import {
NbAdjustableConnectedPositionStrategy,
Expand Down Expand Up @@ -252,10 +252,13 @@ export class NbSelectComponent<T> implements OnInit, AfterViewInit, AfterContent
*/
overlayPosition: NbPosition = '' as NbPosition;

/**
* Stream of events that will fire when one of the options fire selectionChange event.
* */
protected selectionChange$: Subject<NbOptionComponent<T>> = new Subject();
/**
* Stream of events that will fire when one of the options is clicked.
* @deprecated
* Use nb-select (selected) binding to track selection change and <nb-option (click)=""> to track option click.
* @breaking-change 4.0.0
**/
readonly selectionChange: Observable<NbOptionComponent<T>> = this.selectionChange$.asObservable();

protected ref: NbOverlayRef;
Expand Down Expand Up @@ -339,7 +342,7 @@ export class NbSelectComponent<T> implements OnInit, AfterViewInit, AfterContent

this.subscribeOnTriggers();
this.subscribeOnPositionChange();
this.subscribeOnSelectionChange();
this.subscribeOnOptionClick();
}

ngOnDestroy() {
Expand Down Expand Up @@ -395,7 +398,7 @@ export class NbSelectComponent<T> implements OnInit, AfterViewInit, AfterContent
/**
* Selects option or clear all selected options if value is null.
* */
protected handleSelect(option: NbOptionComponent<T>) {
protected handleOptionClick(option: NbOptionComponent<T>) {
if (option.value) {
this.selectOption(option);
} else {
Expand Down Expand Up @@ -505,19 +508,24 @@ export class NbSelectComponent<T> implements OnInit, AfterViewInit, AfterContent
});
}

protected subscribeOnSelectionChange() {
this.subscribeOnOptionsSelectionChange();
protected subscribeOnOptionClick() {
/**
* If the user changes provided options list in the runtime we have to handle this
* and resubscribe on options selection changes event.
* Otherwise, the user will not be able to select new options.
* */
this.options.changes
.subscribe(() => this.subscribeOnOptionsSelectionChange());

this.selectionChange
.pipe(takeWhile(() => this.alive))
.subscribe((option: NbOptionComponent<T>) => this.handleSelect(option));
.pipe(
startWith(this.options),
switchMap((options: QueryList<NbOptionComponent<T>>) => {
return merge(...options.map(option => option.click));
}),
takeWhile(() => this.alive),
)
.subscribe((clickedOption: NbOptionComponent<T>) => {
this.handleOptionClick(clickedOption);
this.selectionChange$.next(clickedOption);
});
}

protected getContainer() {
Expand Down Expand Up @@ -550,20 +558,21 @@ export class NbSelectComponent<T> implements OnInit, AfterViewInit, AfterContent
throw new Error('Can\'t assign array if select is not marked as multiple');
}

this.cleanSelection();
const previouslySelectedOptions = this.selectionModel;
this.selectionModel = [];

if (isArray) {
(<T[]> value).forEach((option: T) => this.selectValue(option));
} else {
this.selectValue(<T> value);
}

this.cd.markForCheck();
}
// find options which were selected before and trigger deselect
previouslySelectedOptions
.filter((option: NbOptionComponent<T>) => !this.selectionModel.includes(option))
.forEach((option: NbOptionComponent<T>) => option.deselect());

protected cleanSelection() {
this.selectionModel.forEach((option: NbOptionComponent<T>) => option.deselect());
this.selectionModel = [];
this.cd.markForCheck();
}

/**
Expand Down Expand Up @@ -591,10 +600,4 @@ export class NbSelectComponent<T> implements OnInit, AfterViewInit, AfterContent
protected isClickedWithinComponent($event: Event) {
return this.hostRef.nativeElement === $event.target || this.hostRef.nativeElement.contains($event.target as Node);
}

protected subscribeOnOptionsSelectionChange() {
merge(...this.options.map(it => it.selectionChange))
.pipe(takeWhile(() => this.alive))
.subscribe((change: NbOptionComponent<T>) => this.selectionChange$.next(change));
}
}
Loading

0 comments on commit 788a6d4

Please sign in to comment.