From Angular 1.x ng-repeat to Angular (2+) NgFor with Component

By Oren Farhi on Jan 29, 2016

In the recent article, I wrote about the migration process of a component that is based on AngularJS.x and ES2015. Since then, I continued migrating Echoes Player to Angular (+2). In this post i’m sharing the migration process of the youtube-list component - which uses the youtube-media component, as well as implementing a smart component with Angular (+2).

The Youtube-List Component

The goal for this process was to reuse the recent youtube-media component and create a youtube-list component. Echoes Player main feature is displaying a list of youtube videos like the following:

Screen Shot 2016-01-29 at 11.18.58 AM

The AngularJS.x ES2015 implementation of youtube-list is a core/component element. It is relatively a minimal component which reuse the youtube-media component. The template uses “ng-repeat” to render a list of videos:

		ng-repeat="video in vm.videos track by $index"

This is the directive definition:

import template from './youtube-list.tpl.html';

// Usage:
//	<youtube-list videos on-select="func(video)" on-queue="func(video)"></youtube-list>
/* @ngInject */
export default function youtubeList() {
	var directive = {
		restrict: 'E',
		replace: true,
		scope: {
			videos: '=',
			onSelect: '&',
			onQueue: '&',
			onAdd: '&'
		bindToController: true,
		controllerAs: 'vm',
		controller: class YoutubeList {
			constructor () {}

			playSelectedVideo (video){
				this.onSelect({ video });

			queueSelectedVideo (video) {
				this.onQueue({ video });

			add (video) {
				this.onAdd({ video });
	return directive;

The Angular (+2) YoutubeList Component

The result of this component in Angular (+2) is not so far from its AngularJS.x version. However, the migration process of this component involves few points to consider.

Importing Relevant Dependencies

With Angular (+2), each dependancy must be defined. That includes:

  1. angular’s 2 built in directives: NgFor (instead of ng-repeat)
  2. the youtube-media component
  3. angular’s 2 annotations & docorators (@Component, @Input etc..)

Let’s import all of these:

import { Component, EventEmitter, Input, Output } from '@angular/core';
import { NgFor } from '@angular/common';
import { YoutubeMedia } from '../youtube-media/youtube-media';

Note: ”NgFor” is the new “ng-repeat” in Angular (+2). More on that below.

Define The Component and its dependencies

This code defines how the youtube-list component will be used, its template, its input and which directives it consumes.

	selector: 'youtube-list',
	template: require('./youtube-list.html'),
	inputs: [
	directives: [NgFor, YoutubeMedia ]

Notice how “NgFor” is injected to this component’s directives array. If it’s not defined as such, Angular (+2) template engine won’t use it.

The YoutubeList Controller Class

The controller class in Angular (+2) of this component, is quite similar to its AngularJS.x version. The noticeable addition is the definition of the events that this component exposes.

export class YoutubeList {
	@Output() play = new EventEmitter();
	@Output() queue = new EventEmitter();
	@Output() add = new EventEmitter();

	constructor () {}

	playSelectedVideo (media) {

	queueSelectedVideo (media) {

	addVideo (media) {

Youtubelist Component Template

This component’s template is quite simple. It repeats the youtube-media component following the “list” property.

	*ngFor="#media of list"

Lets overview the ”NgFor” directive usage in this template.

First, you notice that i’m using the syntactic sugar form of ngFor using the ”*****“. It makes it easier to read (and yes - it’s valid html attribute). This ”*” form is almost similar to the AngularJS.x ”ng-repeat”. The value of this attribute is actually almost similar for ES2015 “for of” loop, however, with little micro-syntax inside.

The ”#” sign, is used to defined local in-context variable that will be used throughout the iteration, which we can reference to in other places inside the template. In this template, the ”media” variable is used to reference each time a different item in the list array.

The exposed events: ”play”, ”queue” and ”add”, pass the relevant ”media” variable that was eventually clicked inside the youtube-media component.

Using the YoutubeList Component

Using this component should be simple. I designed it so I can use it like so:

Notice how the "**play**" event passes the "**$event**" argument - which will eventually be the selected media. This is a very important point to realise. In contrary to the "**ngFor**", there's is no reference in this context to a "**video**" property, but rather only to "**videos**" array. Referencing "**media**" here won't work (as we've probably did in AngularJS.x).

In practice, I created a "**youtube-videos**" smart component - a component that is rendered without any attributes, and it is attached to a certain route (currently the index route). This component initiates an http request call to youtube's api and upon response, saves the result in a "**videos**" property.

## Final Thoughts

You can follow the full code commits of <a href="https://github.com/orizens/echoes-ng2/issues/4" target="_blank">youtube-list commits in github</a>.

<a href="http://github.com/orizens/echoes-ng2" target="_blank">Echoes Player with ng2</a> is an open source project that you can follow, fork and overview at github.

If you're still using AngularJS.x and looking to migrate in the future to Angular (+2), I encourage you to start writing <a href="https://github.com/orizens/angular-es2015-styleguide" target="_blank">AngularJS.x with ES2015 and following a style guide</a>. You can overview the <a href="https://github.com/orizens/echoes/tree/es2015" target="_blank">ES2015 branch of Echoes</a> (at production <a href="http://echoesplayer.netlify.app" target="_blank">http://echoesplayer.netlify.app</a>) to see it in action.

Hi there, I'm Oren Farhi

I'm an Experienced Software Engineer, Front End Tech Lead, focusing on Front End & Software Architecture and creating well formed applications.

Profile Avatar
© 2024, Built by Oren Farhi