Build a Positive News iOS Application Using the Power of Machine Learning

How you can create your own positive news iOS application

Natural Language Processing — iOS

Since the beginning of the coronavirus pandemic, the news has been overwhelmingly negative and somewhat depressing. You open Twitter or your favorite news application, and you’re left with loads and loads of information that can drain your energy.

In a world saturated with information, more and more people are choosing to offer positive news, even if it’s still often confined to a specific section.
For example, The British daily newspaper The Guardian offers “The Upside”, while Fox News, MSN, the HuffPost site, and Yahoo! all have “good news” pages.

The most surprising lesson of this crisis is probably the need for people to access ‘good’ news. Google trends data shows a spike of interesting queries (Figure 1) such as good news or positive news during the months of March and April.

The data shows that this is also the case in multiple regions (Europe, India…).

After seeing this, I decided that I might be able to choose what kinds of information to consume in order to break the chain of negativity. I’m a developer so I thought, why not create an application that’ll select only positive news and remove negative and neutral ones for me?

With recent advances in natural language processing (NLP) and mobile processing, we can select, with a certain level of accuracy, the most positive news and create our own custom feeds.

In this article, I’ll create an iOS application that will consume an API and select the most positive articles.

Overview

  1. Find the right news API
  2. Create our iOS application
  3. Create a decodable to parse the API calls
  4. Customize the feed view
  5. Final results and possible improvements

I have included code in this article where it’s most instructive. Full code and data can be found on my GitHub page. Let’s get started.

This is a look at what the final result will look like:

Find the right news API

In order to get a stream of articles, we need to find a good API that will get us articles from various sources. These articles will be used to create our custom news feed. The simplest one is News API — you can customize the API calls and get articles related to a specific country or/and a specific topic.

I chose to get top and breaking news headlines from the United States curated in an API called “Top headlines from the United States”, but you can choose whatever topic interests you.

You do need to create an account to get an apiKey, and the rest is pretty straightforward. You choose the “News sources” and the subcategory, and the website will then generate a GET URL with the appropriate key and news sources.

You can easily test your link in your browser and understand the structure of the API call:

The API call is structured as such:

  • error: Indicates whether the request was successful or not—“ok” for a successful request, and an “error” for a problem.
  • totalResults: Number of articles
  • articles: An array of articles

Create our iOS application

Create a new “Single View Application” and make sure you choose Storyboard as User Interface.

Now we have our project ready to go. I don’t like using storyboards myself, so the app in this tutorial is built programmatically, which means no buttons or switches to toggle — just pure code.

To follow this method, you’ll have to delete the main.storyboard file and set your SceneDelegate.swift file (Xcode 11 only).

With Xcode 11, you’ll have to change the Info.plist file like so:

You need to delete the “Storyboard Name” in the file, and that’s about it.

Change the SceneDelegate with the following code:

var window: UIWindow?

func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {
    guard let windowScene = (scene as? UIWindowScene) else { return }
        
    window = UIWindow(frame: windowScene.coordinateSpace.bounds)
    window?.windowScene = windowScene
    window?.rootViewController = ViewController()
    window?.makeKeyAndVisible()
}

Create View Controllers

We need two ViewControllers:

  • ViewController():

This is where we’ll set our UICollectionView with all the articles retrieved from the news API.

  • DetailsViewController():

This is where we will show the article’s body.

Create a navigation

  • MainTabBarController():

This is our main navigation, where we’ll create the full navigation structure for our application.

Navigation in iOS is pretty straightforward and easy to implement. Here, I’ve changed some things like the color and icon for the navigation bar (and others) so it looks nice.

Create a decodable to parse the API calls

Starting with Swift 4, we can create a representation of an API call using a struct and conforming it to the Decodable protocol. The protocol only works when you read JSON data.

The API returns a series of elements that could be parsed using two important structures. The first one will mimic the API call upper node with a status, totalResult, and finally articles, which is an array of articles. The other one will abstract a representation of an article, and thus will be used to create the array of articles.

Here the structure of the Article struct:

Now that we have defined the properties of an Article , we create the ultimate API call object Articles:

Both structures conform to the Decodable protocol. And the former is using the first one to create an array of Article. Make sure that the properties’ names are the exact same as the API call.

Customize the feed view

Next we will need to customize our feed by adding a UICollectionView that will host our articles.

Create a UICollectionView :

The news feed will have a stream of articles, we will use UICollectionView() that will be instantiated in ViewController() with the following code:

  • Create an object of type UICollectionView()
  • Set the scroll direction
  • Change the background color
  • Toggle the Collection View to be constrained using auto layout

Then, we need to register the cells and set the data source, as well as set up the ViewController() to conform to the UICollectionView() protocol:

Finally, we need to add the UICollectionView to the subview and set up the layout:

  • Add the collection to the subview of the UIViewController()
  • Change the background color to be compatible with Dark Mode

The final step, which is the conformance to UICollectionViewDataSource and UICollectionViewDelegate, is available in the GitHub repository. In this part we will feed the collection view with the data.

Create a custom UICollectionViewCell :

The collection is made up of cells that are members of the feed, think of it as Medium’s homepage, UICollectionViewCell is an instance of an article, and the homepage is the UICollectionView — each cell will have a title, author name and will also lead to the full article. Thus, the homepage is a ‘Collection’ of articles.

I’m using a package made by Paolo Cuscela called Cards that mimics the iOS app store.

For the sake of simplicity, I manually included the files in the project and created a new Class that conforms to the CardDelegate protocol. The Class is also included in the GitHub repository.

class CustomCell: UICollectionViewCell, CardDelegate {
    
    var card: CardArticle!
    var article: Article!
    
    override init(frame: CGRect) {
        super.init(frame: frame)
        self.card = setupCardTest() as? CardArticle
        self.card.delegate = self
        self.article = Article(author: "", title: "", description: "", url: "", publishedAt: "", urlToImage: "", content: "")
    }
    
    func setupCardTest() -> Card {
        
        let screenSize = UIScreen.main.bounds
        let screenHeight = screenSize.height
        let card = CardArticle(frame: CGRect(x: 30, y: 140, width: self.frame.width - 32 , height: self.frame.height/3))
        self.addSubview(card)
        card.translatesAutoresizingMaskIntoConstraints = false
        
        switch UIScreen.main.nativeBounds.height {
        case 1136:
            //iPhones_5_5s_5c_SE
            card.heightAnchor.constraint(equalToConstant: screenHeight/2.5).isActive = true
            card.topAnchor.constraint(equalTo: self.topAnchor, constant: 15).isActive = true
            card.bottomAnchor.constraint(equalTo: self.bottomAnchor, constant: -20).isActive = true
        case 1334:
            //iPhones_6_6s_7_8
            card.heightAnchor.constraint(equalToConstant: screenHeight/2.5).isActive = true
            card.topAnchor.constraint(equalTo: self.topAnchor, constant: 15).isActive = true
            card.bottomAnchor.constraint(equalTo: self.bottomAnchor, constant: -20).isActive = true
        case 1792:
            //iPhone_XR
            card.heightAnchor.constraint(equalToConstant: screenHeight/3).isActive = true
            card.topAnchor.constraint(equalTo: self.topAnchor, constant: self.frame.height/5 - 30).isActive = true
        case 1920, 2208:
            //iPhones_6Plus_6sPlus_7Plus_8Plus
            card.heightAnchor.constraint(equalToConstant: screenHeight/2.5).isActive = true
        case 2436:
            //iPhones_X_XS
            card.heightAnchor.constraint(equalToConstant: screenHeight/3).isActive = true
            card.topAnchor.constraint(equalTo: self.topAnchor, constant: self.frame.height/5 - 30).isActive = true
        case 2688:
            //iPhone_XSMax
            card.heightAnchor.constraint(equalToConstant: screenHeight/2.5).isActive = true
        default:
            card.heightAnchor.constraint(equalToConstant: screenHeight/4).isActive = true
            card.topAnchor.constraint(equalTo: self.topAnchor, constant: self.frame.height/5 - 30).isActive = true
        }
        
        card.centerXAnchor.constraint(equalTo: self.centerXAnchor).isActive = true
        card.widthAnchor.constraint(equalToConstant: self.frame.width - 32).isActive = true
        card.backgroundColor = .systemBackground
        card.subtitle = ""
        card.title = "titre"
        card.category = "author"
        card.textColor = UIColor.black
        card.hasParallax = false
        return card
    }
    
    override func prepareForReuse() {
        self.card.title = "titre"
        self.card.subtitle = ""
        self.card.category = "author"
        self.card.textColor = UIColor.white
    }
    
    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
}

Here’s a breakdown of the CustomCell() our UICollectionViewCell() :

  • First, we need to instantiate an object of type CardArticle which will be the design structure of the cell
  • Then an object of type Article that contains the data
  • Create an initializer where we initialize the Card and the Article object
  • Create a function that will set up the Card’s bounds and attributes (title, author, background color, text color and many more)
  • Finally, we need to override an important function that gets invoked every time a cell is reused by the UICollectionView() . This function exists to improve the efficiency and the performance of the feed. Thus, when a cell is not visible on the screen, the collection view dequeues it and remove it from the stack.

Predict and populate the feed

Since iOS 13, Apple has included in its NaturalLanguage framework the ability to leverage a built-in solution without any external resource.

This feature, more broadly, is known as sentiment analysis, and it gives you the power to access the general sentiment in a given word, sentence, paragraph, or document. The sentiment score is structured as a float number that can range from -1.0 to 1.0. A score of 1.0 is the most positive, a score of -1.0 is the most negative, and a score of 0.0 is neutral.

Since we only want positive text, we’ll only surface articles that have a score strictly above 0.

Here’s a break down of the code:

  • Instantiate a NLTagger and choose sentimentScore as an option.
  • Set the object’s string by passing the input text.
  • Run inference by calling the method tag(), specifying that the text is a paragraph for better prediction.
  • Parse the output as a Double.
  • Test the output and set the final score if the text is positive (1), negative (0) or neutral (-1).

Before loading the CollectionView, we’ll run inference on each and every article’s text.

Final results and possible improvements

The application calls an API and only shows the most positive articles. Unfortunately, the News API is not free and only returns a portion of articles. Thus, there are a number of possibilities for improving the overall experience by adding or modifying the following elements:

  • Create your own API by scraping and parsing your own favorite news outlet, which will help you customize your feed even more.
  • Create your own custom NLP model using state-of-the-art architectures such as Transformer GPT-2 or BERT. In these cases, the model will perform the inference on the server-side rather than on-device. I wrote an article that can help you get started with sentiment analysis using Transformers.
  • Improve the overall experience by running inference in backend while scraping and parsing articles.
  • Add a section for the most important and breaking news articles in order to stay up to date.
  • Setup a notification system in order to receive intermittent news alerts.
  • Add a share functionality to spread positivity around you 😃

The Internet is full of positive news, you just need to search harder on the various news sites and you will find good news. By using your coding skills you can leverage the power of ML and programming and totally automate the process. The news feed can also cover subjects that you deem important, as long as the news fits your own expectations.

Thanks for reading this article. If you have any questions, don’t hesitate to send me an email at [email protected].

Avatar photo

Fritz

Our team has been at the forefront of Artificial Intelligence and Machine Learning research for more than 15 years and we're using our collective intelligence to help others learn, understand and grow using these new technologies in ethical and sustainable ways.

Comments 0 Responses

Leave a Reply

Your email address will not be published. Required fields are marked *