fix(change_detection): allow to destroy `OnPush` components inside of a host event.
Closes #7192
This commit is contained in:
parent
331b9c1317
commit
ebd438ff5e
|
@ -174,7 +174,7 @@ export class ChangeDetectorJITGenerator {
|
||||||
var evalRecord = this._logic.genEventBindingEvalValue(eb, r);
|
var evalRecord = this._logic.genEventBindingEvalValue(eb, r);
|
||||||
var markPath = this._genMarkPathToRootAsCheckOnce(r);
|
var markPath = this._genMarkPathToRootAsCheckOnce(r);
|
||||||
var prevDefault = this._genUpdatePreventDefault(eb, r);
|
var prevDefault = this._genUpdatePreventDefault(eb, r);
|
||||||
return `${evalRecord}\n${markPath}\n${prevDefault}`;
|
return `${markPath}\n${evalRecord}\n${prevDefault}`;
|
||||||
} else {
|
} else {
|
||||||
return this._logic.genEventBindingEvalValue(eb, r);
|
return this._logic.genEventBindingEvalValue(eb, r);
|
||||||
}
|
}
|
||||||
|
|
|
@ -60,9 +60,11 @@ export class DynamicChangeDetector extends AbstractChangeDetector<any> {
|
||||||
if (proto.isSkipRecord()) {
|
if (proto.isSkipRecord()) {
|
||||||
protoIdx += this._computeSkipLength(protoIdx, proto, values);
|
protoIdx += this._computeSkipLength(protoIdx, proto, values);
|
||||||
} else {
|
} else {
|
||||||
var res = this._calculateCurrValue(proto, values, locals);
|
|
||||||
if (proto.lastInBinding) {
|
if (proto.lastInBinding) {
|
||||||
this._markPathAsCheckOnce(proto);
|
this._markPathAsCheckOnce(proto);
|
||||||
|
}
|
||||||
|
var res = this._calculateCurrValue(proto, values, locals);
|
||||||
|
if (proto.lastInBinding) {
|
||||||
return res;
|
return res;
|
||||||
} else {
|
} else {
|
||||||
this._writeSelf(proto, res, values);
|
this._writeSelf(proto, res, values);
|
||||||
|
|
|
@ -755,6 +755,30 @@ function declareTests() {
|
||||||
async.done();
|
async.done();
|
||||||
})}));
|
})}));
|
||||||
|
|
||||||
|
if (DOM.supportsDOMEvents()) {
|
||||||
|
it("should allow to destroy a component from within a host event handler",
|
||||||
|
inject([TestComponentBuilder], fakeAsync((tcb: TestComponentBuilder) => {
|
||||||
|
|
||||||
|
var fixture: ComponentFixture;
|
||||||
|
tcb.overrideView(
|
||||||
|
MyComp, new ViewMetadata({
|
||||||
|
template: '<push-cmp-with-host-event></push-cmp-with-host-event>',
|
||||||
|
directives: [[[PushCmpWithHostEvent]]]
|
||||||
|
}))
|
||||||
|
|
||||||
|
.createAsync(MyComp)
|
||||||
|
.then(root => { fixture = root; });
|
||||||
|
tick();
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
var cmpEl = fixture.debugElement.children[0];
|
||||||
|
var cmp: PushCmpWithHostEvent = cmpEl.inject(PushCmpWithHostEvent);
|
||||||
|
cmp.ctxCallback = (_) => fixture.destroy();
|
||||||
|
|
||||||
|
expect(() => cmpEl.triggerEventHandler('click', <Event>{})).not.toThrow();
|
||||||
|
})));
|
||||||
|
}
|
||||||
|
|
||||||
it('should not affect updating properties on the component',
|
it('should not affect updating properties on the component',
|
||||||
inject([TestComponentBuilder, AsyncTestCompleter],
|
inject([TestComponentBuilder, AsyncTestCompleter],
|
||||||
(tcb: TestComponentBuilder, async) => {
|
(tcb: TestComponentBuilder, async) => {
|
||||||
|
@ -1995,6 +2019,16 @@ class PushCmpWithRef {
|
||||||
propagate() { this.ref.markForCheck(); }
|
propagate() { this.ref.markForCheck(); }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'push-cmp-with-host-event',
|
||||||
|
host: {'(click)': 'ctxCallback($event)'},
|
||||||
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
||||||
|
template: ''
|
||||||
|
})
|
||||||
|
class PushCmpWithHostEvent {
|
||||||
|
ctxCallback: Function = (_) => {};
|
||||||
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'push-cmp-with-async',
|
selector: 'push-cmp-with-async',
|
||||||
changeDetection: ChangeDetectionStrategy.OnPush,
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
||||||
|
|
|
@ -253,7 +253,7 @@ class _CodegenState {
|
||||||
var evalRecord = _logic.genEventBindingEvalValue(eb, r);
|
var evalRecord = _logic.genEventBindingEvalValue(eb, r);
|
||||||
var markPath = _genMarkPathToRootAsCheckOnce(r);
|
var markPath = _genMarkPathToRootAsCheckOnce(r);
|
||||||
var prevDefault = _genUpdatePreventDefault(eb, r);
|
var prevDefault = _genUpdatePreventDefault(eb, r);
|
||||||
return "${evalRecord}\n${markPath}\n${prevDefault}";
|
return "${markPath}\n${evalRecord}\n${prevDefault}";
|
||||||
} else {
|
} else {
|
||||||
return _logic.genEventBindingEvalValue(eb, r);
|
return _logic.genEventBindingEvalValue(eb, r);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue