Crafting a weather app with Ionic framework and Forecast.io

by - April 02, 2017

Introduction

In this article we're going to build a simple weather forecast app with Ionic framework, an open-source front-end development kit for building cross-platform mobile apps with HTML5 and Javascript.
Forecast.io is web service that provides weather data for a given latitude and longitude.


Installation

Installing Ionic is quick and easy. To get started, I've summarized the installation steps below. You can find additional information here.
First, download the build of Node.js appropriate for your platform and install it, if you don't already have it on your machine, that is. This should install NPM (Node Package Manager) as well.
Ionic runs on Linux, Mac OS X and Windows, but you'll need to use the command line to create the app, add platforms and launch the app in an emulator or a device. For Windows users, git bash is a better alternative to the platform's command prompt. More details can be found right here.
Next, let's install Cordova.
  1. $ npm install -g cordova
Finally, install Ionic.
  1. $ sudo npm install -g gulp ionic
Ionic framework should now be successfully installed on your machine.
Now let's create an app. From the terminal, go to your project folder. Type the following command to create a blank app with tabs.
  1. $ ionic start weather tabs
Simply replace tabs in the above line with blank or sidemenu to create a blank app, or an application with a side menu.
Finally navigate to the project directory and add iOS and/or Android platforms to Ionic and test the app.
  1. $cd weather
  2. $ ionic platform add ios
  3. $ ionic build ios
  4. $ ionic emulate ios
That's it! In a few lines, we've installed Ionic and quickly created a starter app. It doesn't really do much at this time so let's get to work.

Building the app

We're building a simple app that displays current weather data for a selected city by pulling data from Forecast.io.
Our app has three pages. Page one displays the current temperature and weather conditions for a default city. The second page contains a list of cities. Selecting one of these cities will display its weather details. While the third page is a placeholder for storing app settings, we won't be covering them in this tutorial.
Before we begin, head over to Forecast.io and signup as a developer to get your own API key for use with this walkthrough.
The application should look like the image below when we are done.
Screen 1 - Home:
Screenshot 1
Screen 2 - Cities
Screenshot 2
To complete the app we need to:
  • Define the front end views
  • Define controllers that provide data
  • Define routes to tie the views to the controllers
So, let's get started. Navigate to the folder www/templates from your application root directory. This folder contains template files that will be rendered for the app. Since we opted to use the tabs style starter app, Ionic has created some files in this folder.
To make things a bit easier, let's modify the generated files slightly to suit our needs.
  • Delete friend-detail.html
  • Rename tab-dash.html to tab-home.html
  • Rename tab-friends.html to tab-cities.html
  • Rename tab-account.html to tab-settings.html
Replace code in tab-home.html with the code below.
  1. <ion-view title="Home">
  2.   <ion-content class="has-header">
  3.     <!-- Display current location's weather data-->
  4.     <h3>{{city}}</h3>
  5.     <h5><weather-icon icon="current.icon" id="current-icon"></weather-icon> {{current.currently.summary}}</h5>
  6.     <p>The temperature now is</p>  
  7.   <span class="large">{{current.currently.temperature}} ° </span>
  8.   </ion-content>
  9. </ion-view>
Here we're displaying the weather data for the current city. Current is a scope item that contains weather data returned from Forecast.io.
Save your changes. Next, open tab-cities.html and replace the code inside it with the code below to display a list of cities using ion-list.
  1. <ion-view title="Cities">
  2.   <ion-content class="has-header">
  3.       <h1>Select city</h1>
  4.         <ion-list>
  5.               <ion-item ng-repeat="city in cities">
  6.               <span href="#" ng-click="changeCity('{{city.id}}')">
  7.                 {{city.name}}
  8.               </span>
  9.               </ion-item>
  10.         </ion-list>
  11.    </ion-content>
  12. </ion-view>
Next open tabs.html. This contains code that defines the tabs that appear at the bottom of all pages. Replace the code inside the file with the code below to link the tabs with the view templates we just modified.
  1. <ion-tabs class="tabs-icon-top">
  2.   <!-- home Tab -->
  3.   <ion-tab title="Home" icon="icon ion-home" href="#/tab/home">
  4.     <ion-nav-view name="tab-home"></ion-nav-view>
  5.   </ion-tab>
  6.   <!-- Cities Tab -->
  7.   <ion-tab title="Change City" icon="icon ion-heart" href="#/tab/city">
  8.     <ion-nav-view name="tab-cities"></ion-nav-view>
  9.   </ion-tab>
  10.   <!-- Settings Tab -->
  11.   <ion-tab title="Settings" icon="icon ion-gear-b" href="#/tab/settings">
  12.     <ion-nav-view name="tab-settings"></ion-nav-view>
  13.   </ion-tab>
  14. </ion-tabs>
Next, let's define the routes. Open js/app.js and replace the code with the lines below.
  1. angular.module('starter', ['ionic', 'starter.controllers', 'starter.services'])
  2. .run(function($ionicPlatform) {
  3.   $ionicPlatform.ready(function() {
  4.     // Hide the accessory bar by default (remove this to show the accessory bar above the keyboard
  5.     // for form inputs)
  6.     if(window.cordova && window.cordova.plugins.Keyboard) {
  7.       cordova.plugins.Keyboard.hideKeyboardAccessoryBar(true);
  8.     }
  9.     if(window.StatusBar) {
  10.       // org.apache.cordova.statusbar required
  11.       StatusBar.styleDefault();
  12.     }
  13.   });
  14. })
  15. .config(function($stateProvider, $urlRouterProvider) {
  16. $stateProvider
  17. // setup an abstract state for the tabs directive
  18.     .state('tab', {
  19.       url: "/tab",
  20.       abstract: true,
  21.       templateUrl: "templates/tabs.html"
  22.     })
  23.     // Each tab has its own nav history stack:
  24.     .state('tab.home', {
  25.       url: '/home',
  26.       views: {
  27.         'tab-home': {
  28.           templateUrl: 'templates/tab-home.html',
  29.           controller: 'HomeCtrl'
  30.         }
  31.       }
  32.     })
  33.     .state('tab.changecity', {
  34.       url: '/city',
  35.       views: {
  36.         'tab-cities': {
  37.           templateUrl: 'templates/tab-cities.html',
  38.           controller: 'LocationsCtrl'
  39.         }
  40.       }
  41.     })
  42.     .state('tab.settings', {
  43.       url: '/settings',
  44.       views: {
  45.         'tab-settings': {
  46.           templateUrl: 'templates/tab-settings.html',
  47.           controller: 'SettingsCtrl'
  48.         }
  49.       }
  50.     });
  51. // if none of the above states are matched, use this as the fallback
  52.   $urlRouterProvider.otherwise('/tab/home');
  53. });
After defining templates and adding routes, we'll create controllers. Open js/controllers.jsand replace the code inside it with what's below.
  1. angular.module('starter.controllers', ['ionic'])
  2. .constant('FORECASTIO_KEY', 'your forecastio key here')
  3. .controller('HomeCtrl', function($scope,$state,Weather,DataStore) {
  4.     //read default settings into scope
  5.     console.log('inside home');
  6.     $scope.city  = DataStore.city;
  7.     var latitude  =  DataStore.latitude;
  8.     var longitude = DataStore.longitude;
  9.     //call getCurrentWeather method in factory ‘Weather’
  10.     Weather.getCurrentWeather(latitude,longitude).then(function(resp) {
  11.       $scope.current = resp.data;
  12.       console.log('GOT CURRENT', $scope.current);
  13.       //debugger;
  14.     }, function(error) {
  15.       alert('Unable to get current conditions');
  16.       console.error(error);
  17.     });
  18. })
  19. .controller('LocationsCtrl', function($scope,$state, Cities,DataStore) {
  20.   $scope.cities = Cities.all();
  21.   $scope.changeCity = function(cityId) {
  22.     //get lat and longitude for seleted location
  23.     var lat  = $scope.cities[cityId].lat; //latitude
  24.     var lgn  = $scope.cities[cityId].lgn; //longitude
  25.     var city = $scope.cities[cityId].name; //city name
  26.     DataStore.setCity(city);
  27.     DataStore.setLatitude(lat);
  28.     DataStore.setLongitude(lgn);
  29.     $state.go('tab.home');
  30.   }
  31. })
  32. .controller('SettingsCtrl', function($scope) {
  33.     //manages app settings
  34. });
In the code above, we define three controllers:
  • Home: Defines two scopes, city and current. Current is data returned from an API request. City names are retrieved from DataStore factory. Here we're calling the method 'getCurrentWeather' defined in factory 'Weather' passing in latitude and longitude.
  • Locations: Controls city view
  • Settings: Controls the settings view
In the home and locations controllers, we inject three factories:
  • Cities
  • DataStore: Stores data used across controllers
  • Weather: Contains methods to access the Forecast.io API for weather data
Lastly, let's take a look at js/services.js. This file defines services and factory objects. Factory objects allow different controllers to share the same data.
  1. 'use strict';
  2. var forecastioWeather = ['$q', '$resource', '$http', 'FORECASTIO_KEY',
  3.   function($q, $resource, $http, FORECASTIO_KEY) {
  4.   var url = 'https://api.forecast.io/forecast/' + FORECASTIO_KEY + '/';
  5.   var weatherResource = $resource(url, {
  6.     callback: 'JSON_CALLBACK',
  7.   }, {
  8.     get: {
  9.       method: 'JSONP'
  10.     }
  11.   });
  12.   return {
  13.     //getAtLocation: function(lat, lng) {
  14.     getCurrentWeather: function(lat, lng) {
  15.       return $http.jsonp(url + lat + ',' + lng + '?callback=JSON_CALLBACK');
  16.     }
  17.   }
  18. }];
  19. angular.module('starter.services', ['ngResource'])
  20. .factory('Cities', function() {
  21. var cities = [
  22.     { id: 0, name: 'Miami', lat:25.7877 , lgn: 80.2241 },
  23.     { id: 1, name: 'New York City' ,lat: 40.7127 , lgn: 74.0059 },
  24.     { id: 2, name: 'London' ,lat:51.5072 , lgn: 1.1275 },
  25.     { id: 3, name: 'Los Angeles' ,lat: 34.0500 , lgn: 118.2500 },
  26.     { id: 4, name: 'Dallas' ,lat: 32.7758 , lgn:96.7967  },
  27.     { id: 5, name: 'Frankfurt' ,lat:50.1117 , lgn: 8.6858 },
  28.     { id: 6, name: 'New Delhi' ,lat:28.6100 , lgn: 77.2300 }
  29.   ];
  30.   return {
  31.     all: function() {
  32.       return cities;
  33.     },
  34.     get: function(cityId) {
  35.       // Simple index lookup
  36.       return cities[cityId];
  37.     }
  38.   }
  39. }).
  40. factory('DataStore', function() {
  41.     //create datastore with default values
  42.     var DataStore = {
  43.         city:       'Miami',
  44.         latitude:   25.7877,
  45.         longitude:  80.2241 };
  46.     DataStore.setCity = function (value) {
  47.        DataStore.city = value;
  48.     };
  49.     DataStore.setLatitude = function (value) {
  50.        DataStore.longitude = value;
  51.     };
  52.     DataStore.setLongitude = function (value) {
  53.        DataStore.longitude = value;
  54.     };
  55.     return DataStore;
  56. })
  57. .factory('Weather', forecastioWeather);
We have to make on last change before we can see the app in action. Navigate to the "www" folder and open index.html.
Next, locate the line containing
<script src="lib/ionic/js/ionic.bundle.js"></script>
Now, add the following after the line above.
<script src="lib/ionic/js/angular/angular-resource.js"></script>
That's it. Save the file, and build and run the app from the command prompt:
  1. $ionic build ios
  2. $ionic run ios
ionic run ios installs the build on an ios device. If you don't have a device on hand, you can run the app in an emulator with the following command:
$ionic emulate ios
You can also view the app in a browser. If you don't have web server installed, you can use Python's SimpleHTTPServer. From the terminal, navigate to the "www" folder and run the following command.
$python -m SimpleHTTPServer 3000
Now, open your favorite browser and visit http://localhost:3000 to see the app running.

Conclusion

In this article, we saw how to get started with Ionic, an advanced HTML5 framework for hybrid app development.
In future articles, we'll take a look at enhancing the app by adding support for a remote backend with Parse.
If you're unfamiliar with Angular, I recommended taking a look at the AngularJS docs for a better understanding of it. In the mean time, you can get this article's source code on GitHub
Happy coding!

You May Also Like

0 comments