fix(ivy): support ICU expressions inserted in ngTemplateOutlets inside ngFors (#31789)

This commit fixes a bug where ICU expressions inserted into ngTemplateOutlets
that are inside ngFor blocks would throw an error. We were assuming in view
insertion code that text nodes would always exist by the time a view\`s
creation block had executed. This is not true for text nodes created dynamically
by ICUs because this happens in the update block (in `i18nApply`).

This change ensures such dynamically created nodes are skipped when encountered
too early (as they will be attached later by i18n code anyway).

PR Close #31789
This commit is contained in:
Kara Erickson 2019-07-22 20:55:07 -07:00
parent 215ef3c5f4
commit 54ef63b0f4
2 changed files with 63 additions and 25 deletions

View File

@ -70,32 +70,38 @@ const enum WalkTNodeTreeAction {
function executeActionOnElementOrContainer( function executeActionOnElementOrContainer(
action: WalkTNodeTreeAction, renderer: Renderer3, parent: RElement | null, action: WalkTNodeTreeAction, renderer: Renderer3, parent: RElement | null,
lNodeToHandle: RNode | LContainer | LView, beforeNode?: RNode | null) { lNodeToHandle: RNode | LContainer | LView, beforeNode?: RNode | null) {
ngDevMode && assertDefined(lNodeToHandle, '\'lNodeToHandle\' is undefined'); // If this slot was allocated for a text node dynamically created by i18n, the text node itself
let lContainer: LContainer|undefined; // won't be created until i18nApply() in the update block, so this node should be skipped.
let isComponent = false; // For more info, see "ICU expressions should work inside an ngTemplateOutlet inside an ngFor"
// We are expecting an RNode, but in the case of a component or LContainer the `RNode` is wrapped // in `i18n_spec.ts`.
// in an array which needs to be unwrapped. We need to know if it is a component and if if (lNodeToHandle != null) {
// it has LContainer so that we can process all of those cases appropriately. let lContainer: LContainer|undefined;
if (isLContainer(lNodeToHandle)) { let isComponent = false;
lContainer = lNodeToHandle; // We are expecting an RNode, but in the case of a component or LContainer the `RNode` is
} else if (isLView(lNodeToHandle)) { // wrapped
isComponent = true; // in an array which needs to be unwrapped. We need to know if it is a component and if
ngDevMode && assertDefined(lNodeToHandle[HOST], 'HOST must be defined for a component LView'); // it has LContainer so that we can process all of those cases appropriately.
lNodeToHandle = lNodeToHandle[HOST] !; if (isLContainer(lNodeToHandle)) {
} lContainer = lNodeToHandle;
const rNode: RNode = unwrapRNode(lNodeToHandle); } else if (isLView(lNodeToHandle)) {
ngDevMode && assertDomNode(rNode); isComponent = true;
ngDevMode && assertDefined(lNodeToHandle[HOST], 'HOST must be defined for a component LView');
lNodeToHandle = lNodeToHandle[HOST] !;
}
const rNode: RNode = unwrapRNode(lNodeToHandle);
ngDevMode && assertDomNode(rNode);
if (action === WalkTNodeTreeAction.Insert) { if (action === WalkTNodeTreeAction.Insert) {
nativeInsertBefore(renderer, parent !, rNode, beforeNode || null); nativeInsertBefore(renderer, parent !, rNode, beforeNode || null);
} else if (action === WalkTNodeTreeAction.Detach) { } else if (action === WalkTNodeTreeAction.Detach) {
nativeRemoveNode(renderer, rNode, isComponent); nativeRemoveNode(renderer, rNode, isComponent);
} else if (action === WalkTNodeTreeAction.Destroy) { } else if (action === WalkTNodeTreeAction.Destroy) {
ngDevMode && ngDevMode.rendererDestroyNode++; ngDevMode && ngDevMode.rendererDestroyNode++;
(renderer as ProceduralRenderer3).destroyNode !(rNode); (renderer as ProceduralRenderer3).destroyNode !(rNode);
} }
if (lContainer != null) { if (lContainer != null) {
executeActionOnContainer(renderer, action, lContainer, parent, beforeNode); executeActionOnContainer(renderer, action, lContainer, parent, beforeNode);
}
} }
} }

View File

@ -921,6 +921,38 @@ onlyInIvy('Ivy i18n logic').describe('runtime i18n', () => {
expect(fixture.debugElement.nativeElement.innerHTML).not.toContain('A - Type A'); expect(fixture.debugElement.nativeElement.innerHTML).not.toContain('A - Type A');
expect(fixture.debugElement.nativeElement.innerHTML).toContain('other - Type C'); expect(fixture.debugElement.nativeElement.innerHTML).toContain('other - Type C');
}); });
it('should work inside an ngTemplateOutlet inside an ngFor', () => {
@Component({
selector: 'app',
template: `
<ng-template #myTemp i18n let-type>{
type,
select,
A {A }
B {B }
other {other - {{ typeC // i18n(ph="PH WITH SPACES") }}}
}
</ng-template>
<div *ngFor="let type of types">
<ng-container *ngTemplateOutlet="myTemp; context: {$implicit: type}">
</ng-container>
</div>
`
})
class AppComponent {
types = ['A', 'B', 'C'];
}
TestBed.configureTestingModule({declarations: [AppComponent]});
const fixture = TestBed.createComponent(AppComponent);
fixture.detectChanges();
expect(fixture.debugElement.nativeElement.innerHTML).toContain('A');
expect(fixture.debugElement.nativeElement.innerHTML).toContain('B');
});
}); });
describe('should support attributes', () => { describe('should support attributes', () => {