/** * @license * Copyright Google Inc. All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import {EventEmitter} from '@angular/core'; import {C, E, L, LifecycleHook, T, V, b, cR, cr, defineComponent, defineDirective, e, l, p, v} from '../../src/render3/index'; import {containerEl, renderToHtml} from './render_util'; describe('outputs', () => { let buttonToggle: ButtonToggle; class ButtonToggle { change = new EventEmitter(); resetStream = new EventEmitter(); static ngComponentDef = defineComponent({ tag: 'button-toggle', template: function(ctx: any, cm: boolean) {}, factory: () => buttonToggle = new ButtonToggle(), outputs: {change: 'change', resetStream: 'reset'} }); } let otherDir: OtherDir; class OtherDir { changeStream = new EventEmitter(); static ngDirectiveDef = defineDirective( {factory: () => otherDir = new OtherDir, outputs: {changeStream: 'change'}}); } it('should call component output function when event is emitted', () => { /** */ function Template(ctx: any, cm: boolean) { if (cm) { E(0, ButtonToggle); { L('change', ctx.onChange.bind(ctx)); } e(); } ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); } let counter = 0; const ctx = {onChange: () => counter++}; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(counter).toEqual(1); buttonToggle !.change.next(); expect(counter).toEqual(2); }); it('should support more than 1 output function on the same node', () => { /** */ function Template(ctx: any, cm: boolean) { if (cm) { E(0, ButtonToggle); { L('change', ctx.onChange.bind(ctx)); L('reset', ctx.onReset.bind(ctx)); } e(); } ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); } let counter = 0; let resetCounter = 0; const ctx = {onChange: () => counter++, onReset: () => resetCounter++}; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(counter).toEqual(1); buttonToggle !.resetStream.next(); expect(resetCounter).toEqual(1); }); it('should eval component output expression when event is emitted', () => { /** */ function Template(ctx: any, cm: boolean) { if (cm) { E(0, ButtonToggle); { L('change', () => ctx.counter++); } e(); } ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); } const ctx = {counter: 0}; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(ctx.counter).toEqual(1); buttonToggle !.change.next(); expect(ctx.counter).toEqual(2); }); it('should unsubscribe from output when view is destroyed', () => { /** * % if (condition) { * * % } */ function Template(ctx: any, cm: boolean) { if (cm) { C(0); } cR(0); { if (ctx.condition) { if (V(0)) { E(0, ButtonToggle); { L('change', ctx.onChange.bind(ctx)); } e(); } ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); v(); } } cr(); } let counter = 0; const ctx = {onChange: () => counter++, condition: true}; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(counter).toEqual(1); ctx.condition = false; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(counter).toEqual(1); }); it('should unsubscribe from output in nested view', () => { /** * % if (condition) { * % if (condition2) { * * % } * % } */ function Template(ctx: any, cm: boolean) { if (cm) { C(0); } cR(0); { if (ctx.condition) { if (V(0)) { C(0); } cR(0); { if (ctx.condition2) { if (V(0)) { E(0, ButtonToggle); { L('change', ctx.onChange.bind(ctx)); } e(); } ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); v(); } } cr(); v(); } } cr(); } let counter = 0; const ctx = {onChange: () => counter++, condition: true, condition2: true}; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(counter).toEqual(1); ctx.condition = false; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(counter).toEqual(1); }); it('should work properly when view also has listeners and destroys', () => { let destroyComp: DestroyComp; class DestroyComp { events: string[] = []; ngOnDestroy() { this.events.push('destroy'); } static ngComponentDef = defineComponent({ tag: 'destroy-comp', template: function(ctx: any, cm: boolean) {}, factory: () => { destroyComp = new DestroyComp(); l(LifecycleHook.ON_DESTROY, destroyComp, destroyComp.ngOnDestroy); return destroyComp; } }); } /** * % if (condition) { * * * * % } */ function Template(ctx: any, cm: boolean) { if (cm) { C(0); } cR(0); { if (ctx.condition) { if (V(0)) { E(0, 'button'); { L('click', ctx.onClick.bind(ctx)); T(1, 'Click me'); } e(); E(2, ButtonToggle); { L('change', ctx.onChange.bind(ctx)); } e(); E(4, DestroyComp); e(); } ButtonToggle.ngComponentDef.h(3, 2); DestroyComp.ngComponentDef.h(5, 4); ButtonToggle.ngComponentDef.r(3, 2); DestroyComp.ngComponentDef.r(5, 4); v(); } } cr(); } let clickCounter = 0; let changeCounter = 0; const ctx = {condition: true, onChange: () => changeCounter++, onClick: () => clickCounter++}; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(changeCounter).toEqual(1); expect(clickCounter).toEqual(0); const button = containerEl.querySelector('button'); button !.click(); expect(changeCounter).toEqual(1); expect(clickCounter).toEqual(1); ctx.condition = false; renderToHtml(Template, ctx); expect(destroyComp !.events).toEqual(['destroy']); buttonToggle !.change.next(); button !.click(); expect(changeCounter).toEqual(1); expect(clickCounter).toEqual(1); }); it('should fire event listeners along with outputs if they match', () => { let buttonDir: MyButton; /** */ class MyButton { click = new EventEmitter(); static ngDirectiveDef = defineDirective({factory: () => buttonDir = new MyButton, outputs: {click: 'click'}}); } function Template(ctx: any, cm: boolean) { if (cm) { E(0, 'button', null, [MyButton]); { L('click', ctx.onClick.bind(ctx)); } e(); } } let counter = 0; renderToHtml(Template, {counter, onClick: () => counter++}); // To match current Angular behavior, the click listener is still // set up in addition to any matching outputs. const button = containerEl.querySelector('button') !; button.click(); expect(counter).toEqual(1); buttonDir !.click.next(); expect(counter).toEqual(2); }); it('should work with two outputs of the same name', () => { /** */ function Template(ctx: any, cm: boolean) { if (cm) { E(0, ButtonToggle, null, [OtherDir]); { L('change', ctx.onChange.bind(ctx)); } e(); } ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); } let counter = 0; renderToHtml(Template, {counter, onChange: () => counter++}); buttonToggle !.change.next(); expect(counter).toEqual(1); otherDir !.changeStream.next(); expect(counter).toEqual(2); }); it('should work with an input and output of the same name', () => { let otherDir: OtherDir; class OtherDir { change: boolean; static ngDirectiveDef = defineDirective({factory: () => otherDir = new OtherDir, inputs: {change: 'change'}}); } /** */ function Template(ctx: any, cm: boolean) { if (cm) { E(0, ButtonToggle, null, [OtherDir]); { L('change', ctx.onChange.bind(ctx)); } e(); } p(0, 'change', b(ctx.change)); ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); } let counter = 0; renderToHtml(Template, {counter, onChange: () => counter++, change: true}); expect(otherDir !.change).toEqual(true); renderToHtml(Template, {counter, onChange: () => counter++, change: false}); expect(otherDir !.change).toEqual(false); buttonToggle !.change.next(); expect(counter).toEqual(1); }); it('should work with outputs at same index in if block', () => { /** * // outputs: null * % if (condition) { * // outputs: {change: [0, 'change']} * % } else { *
// outputs: {change: [0, * 'changeStream']} * % } */ function Template(ctx: any, cm: boolean) { if (cm) { E(0, 'button'); { L('click', ctx.onClick.bind(ctx)); T(1, 'Click me'); } e(); C(2); } cR(2); { if (ctx.condition) { if (V(0)) { E(0, ButtonToggle); { L('change', ctx.onChange.bind(ctx)); } e(); } ButtonToggle.ngComponentDef.h(1, 0); ButtonToggle.ngComponentDef.r(1, 0); v(); } else { if (V(1)) { E(0, 'div', null, [OtherDir]); { L('change', ctx.onChange.bind(ctx)); } e(); } v(); } } cr(); } let counter = 0; const ctx = {condition: true, onChange: () => counter++, onClick: () => {}}; renderToHtml(Template, ctx); buttonToggle !.change.next(); expect(counter).toEqual(1); ctx.condition = false; renderToHtml(Template, ctx); expect(counter).toEqual(1); otherDir !.changeStream.next(); expect(counter).toEqual(2); }); });