【发布时间】:2019-11-13 11:36:57
【问题描述】:
我为我的代码创建了一个Stackblitz(链接:https://stackblitz.com/edit/angular-iah7up)。这是我的自定义组件,它根据点击和点击切换两个 ng-content。
import {Component, ElementRef, EventEmitter, OnDestroy, OnInit, Output} from '@angular/core';
import {fromEvent, Subject} from "rxjs";
import {untilDestroyed} from "ngx-take-until-destroy";
import {filter, switchMapTo, take} from "rxjs/operators";
@Component({
selector: 'app-editable-inplace',
templateUrl: './editable-inplace.component.html',
styleUrls: ['./editable-inplace.component.css']
})
export class EditableInplaceComponent implements OnInit, OnDestroy {
@Output() update = new EventEmitter();
mode: 'view' | 'edit' = 'view';
editMode = new Subject();
editMode$ = this.editMode.asObservable();
ngOnInit(): void {
this.viewModeHandler();
this.editModeHandler();
}
constructor(private host: ElementRef) {
}
// This method must be present, even if empty.
ngOnDestroy() {
// To protect you, we'll throw an error if it doesn't exist.
}
private viewModeHandler() {
fromEvent(this.element, 'click').pipe(
untilDestroyed(this)
).subscribe(() => {
console.log('clicked inside')
this.editMode.next(true);
this.mode = 'edit';
});
}
private editModeHandler() {
const clickOutside$ = fromEvent(document, 'click').pipe(
filter(({target}) => {
console.log(this.mode)
console.log('parent', this.element, 'child', target)
const ans = this.element.contains(target) === false
console.log('clickoutside', ans)
return ans
}),
take(1)
)
this.editMode$.pipe(
switchMapTo(clickOutside$),
untilDestroyed(this)
).subscribe(event => {
this.update.next();
this.mode = 'view';
});
}
get element() {
return this.host.nativeElement;
}
toViewMode() {
this.update.next();
this.mode = 'view';
}
}
模板:
<div *ngIf="mode==='view'" >
<ng-content select="[view]"></ng-content>
</div>
<div *ngIf="mode==='edit'" >
<ng-content select="[edit]"></ng-content>
</div>
用法:
<app-editable-inplace >
<div view>
<h3>Click to edit [not working :-( ]</h3>
</div>
<div edit>
<input placeholder="Click to edit" >
</div>
</app-editable-inplace>
但是当我点击视图时,clickOutside$ 被立即触发(我不知道为什么)
而且,this.element.contains(target) === false 行不起作用,因为我可以在控制台中看到主机包含点击的项目,尽管总是说我点击了外部(我也不知道为什么)
【问题讨论】:
标签: angular typescript rxjs