Skip to content

Latest commit

 

History

History
361 lines (243 loc) · 13.4 KB

README.md

File metadata and controls

361 lines (243 loc) · 13.4 KB

CommuteStream iOS Native Ads SDK

This SDK allows you to add CommuteStream Native Ads to your app.

These instructions assume you have already followed the instructions at: https://commutestream.com/sdkinstructions.

Requirements

Downloading

Latest CS iOS Native SDK: http://download.commutestream.com/native-sdk/CSNative-iOS-1.3.0.framework.zip

GitHub: https://github.com/CommuteStream/cs-ios-native-sdk.

Adding the SDK

  1. Drag the CSNative.framework file into your Xcode project.

  2. In your project's target, under Build Phases > Embed Frameworks, drag CSNative.framework to the list of embedded frameworks.

  3. Import the header files:

    Objective-C

    Add #import <CSNative/CSNative.h> to the top of each ViewController that requests ads.

    Swift

    Add #import <CSNative/CSNative.h> to your projects Bridging-Header.h file.

  4. Depending on your project settings, you may need to set Allow Non-modular Includes In Framework Modules to YES in Build Settings.

Overview

CommuteStream Native Ads were built specifically for transit apps. Our SDK is packed with flexible components that allow publishers to seamlessly display advertising alongside public transit information.

Available Components

Components are meant to be used where the publisher sees fit, but should be placed with the understanding that the ad data they display, will be relevant to the transit data surrounding them. The following components are available and listed in order of usage priority:

  • Headline - Displays a strong call-to-action or phrase that relates the value of the offer in as few words as possible –e.g., "Free 12 oz. Coffee"

  • Body - Provides a more detailed description of the offer –e.g., "With purchase any breakfast item."

  • Logo - Shows the brand of the company being advertised. In most cases, logos are meant to be displayed with other components such as a Headline, Body or Advertiser, but can be placed alone as an embeleshment to a list item for a transit stop or route.

  • Advertiser - Shows the name of the advertiser. –e.g., "Just Coffee"

Action Cards

The Action Card is shown anytime a Logo, Body, Headline, Advertiser, or any grouping thereof is tapped on. It is the ultimate native ad, floating above the contents of the app, it keeps the user experience and related transit information in context. The action card includes the following components by default:

  • Title - Explains the transit context of the ad. For example, if the user tapped on an icon that was placed on a train stop list, it might read "Near Belmont Red Line Stop."

  • Subtitle - Further explains the transit context with location information such as a specific address, cross streets, or how far away it is to walk.

  • Hero Image - A large image that reenforces the advertiser's brand or message.

  • Headline, Body, Logo, and Advertiser - the same compenents used by the publisher to entice users to tap and reveal the Action Card.

  • Action Buttons - A maximum of three buttons that perform link-based actions specified by the advertiser. –e.g., "Download App", "Apply Now", or "Learn More"

  • Close Button - An easy way for the user to quickly opt out.

Placing Ad Components

You can add components to your app either programmatically, or by using the Storyboard feature in XCode:

Programmatically

Here is an example of placing components inside a table view cell:

Swift

let logoView = CSNLogoView.init(frame: CGRect(x: 0, y: 0, width: 44, height: 44))
cell.contentView.addSubview(logoView)

let headlineView = CSNHeadlineView.init(frame: CGRect(x: 55, y: 12, width: 200, height: 50))
cell.contentView.addSubview(headlineView)

Objective-C

CSNLogoView *logo = [[CSNLogoView alloc] initWithFrame: CGRectMake(0, 0, 44, 44)];
[cell addSubview:logo];

CSNHeadlineView *headline = [[CSNHeadlineView alloc] initWithFrame: CGRectMake(55, 0, 200, 50)];
[cell addSubview:headline];

Using Storyboards

Some developers prefer to layout components using loadable .xib files. Our SDK makes this simple and intuitive. Use the following UIKit components as they relate to the CS components you want in your .xib file:
  • CSNHeadlineView -> UILabel
  • CSNBodyView -> UILabel
  • CSNLogoView -> UIImageView
  • CSNAdvertiserView -> UILabel

Once you have a UIKit component placed, select it, and view its properties in the right panel's Identity Inspector. Under Custom Class, set its Class property to the corresponding CSNative class from the list above.

For more tips and code snippets, visit our Native Ads Best Practices page.

Building An AdRequest

In previous versions of our SDK, the fetchAds would return a complete list of ads that matched the order of your AdRequest. This functionality remains, however, we have optimized the SDK so that you may display ads as they are returned from the server. This new and preferred method is outlined below.

CSNAdRequest

Once you've placed the components where you want them, you'll need to request a list of native ads to populate them. For this you will create an individual CSNAdRequest, packaged with transit information that matches what's showing on the screen.

There are three levels of transit data you can provide with your CSNAdRequest. Each level is more specific than the last and uses a special class.

IMPORTANT: Be sure to attach a completed block to each CSNAdRequest for handling each ad as it's returned from the server.

CSNTransitAgency (least specific)

Swift

let agencyAdRequest = CSNAdRequest.init()
let agency = CSNTransitAgency.init(ids:"agency_id")

agencyAdRequest.addAgency(agency)
agencyAdRequest.completed = {(ad) in

            //handle returned ad in this block
        }
        

Objective-C

CSNAdRequest *agencyAdRequest = [[CSNAdRequest alloc] init];
CSNTransitAgency *agency = [[CSNTransitAgency alloc] initWithIDs:@"agency_id"];

[agencyAdRequest addAgency:agency];
agencyAdRequest.completed = ^(CSNOptionalAd *ad) {

          //handle returned ad in this block
          }

CSNTransitRoute (more specific)

Swift

let routeAdRequest = CSNAdRequest.init()
let route = CSNTransitRoute.init(ids:"agency_id", routeID:"route_id")


routeAdRequest.addRoute(route)
routeAdRequest.completed = {(ad) in

            //handle returned ad in this block
        }

Objective-C

CSNAdRequest *routeAdRequest = [[CSNAdRequest alloc] init];
CSNTransitRoute *route = [[CSNTransitRoute alloc] initWithIDs:@"agency_id" routeID:@"route_id"];

[routeAdRequest addRoute:route];
routeAdRequest.completed = ^(CSNOptionalAd *ad) {

          //handle returned ad in this block
          });

CSNTransitStop (most specific)

Swift

let stopAdRequest = CSNAdRequest.init()
let stop = CSNTransitStop.init(ids:"agency_id", routeID: "route_id", stopID: "stop_id")

stopAdRequest.addStop(stop)
stopAdRequest.completed = {(ad) in

            //handle returned ad in this block
        }

Objective-C

CSNAdRequest *stopAdRequest = [[CSNAdRequest alloc] init];
CSNTransitStop *stop = [[CSNTransitStop alloc] initWithIDs:@"agency_id" routeID:@"route_id" stopID:@"stop_id"];

[stopAdRequest addStop:stop];
stopAdRequest.completed = ^(CSNOptionalAd *ad) {

          //handle returned ad in this block
          });

Each CSNAdRequest must be added to a mutable array, which will then be used to fetch ads using the CSNAdsController class.

Swift

var adRequests = [CSNAdRequest]()
adRequests.append(adRequest)

Objective-C

NSMutableArray *adRequestsArray = [NSMutableArray alloc] init];
[adRequestsArray addObject:adRequest];

CSNAdsController

In order to fetch ads with the array of CSNAdRequests, you must call the fetchAds method on an instance of CSNAdsController. This will invoke a callback that returns an ads array.

The ads array has the same object count and order as the adRequests array provided in the call. This makes it easy to map to the transit context you made the requests for.

All indexes in the array have a CSNOptionalAd, and each one of these has an ad property that is either a CSNAd or a nil value. You will use this array to display ad content in the components you added to your app design.

Swift

var adsController = CSNAdsController(adUnit: "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");
adsController.fetchAds(adRequests, completed: {(ads) in
	self.stopAds = ads
    self.routeStopsListTableView.reloadData()
})

Objective-C

CSNAdsController *adsController = [[CSNAdsController alloc] initWithAdUnit: @"xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"]
[adsController adsController fetchAds:adRequests completed: ^(NSArray <CSNOptionalAd *> *ads) {
	_ads = ads;
    NSLog(@"%ld", [_ads count]);
    [_tableView reloadData];
}];

Building Ads

Once you've placed your ad components and retreived ads using the fetchAds method, you can put it all together by filling the components with ad data using the buildView method of the CSNAdsController class.

The buildView method accepts three parameters:

  • the parent view of the ad components
  • the ad property of the CSNOptionalAd that was returned in the fetchAds call
  • a boolean value for whether or not the parent is clickable (allows for grouping components so they respond to the same tap)

Objective-C

[adsController buildView:[cell contentView] ad:ad parentTouch:false];

Swift

adController.build(cell.contentView, ad:(ad as CSNOptionalAd).ad, parentTouch: false)

Releasing Your App

Since the .framework file must be embedded into your app, the unnecessary architectures have to be “stripped out." Follow the steps before you archive and upload to the App Store:

  1. Make sure you've added the CSNative.framework to Embed Frameworks in Build Phases.

  2. Also in Build Phases, click the + sign to add a New Run Script Phase. Make sure this is after the Embed Frameworks section.

  3. Set the Shell path to /bin/sh

  4. Paste in the following script:

APP_PATH="${TARGET_BUILD_DIR}/${WRAPPER_NAME}"

# This script loops through the frameworks embedded in the application and
# removes unused architectures.
find "$APP_PATH" -name '*.framework' -type d | while read -r FRAMEWORK
do
    FRAMEWORK_EXECUTABLE_NAME=$(defaults read "$FRAMEWORK/Info.plist" CFBundleExecutable)
    FRAMEWORK_EXECUTABLE_PATH="$FRAMEWORK/$FRAMEWORK_EXECUTABLE_NAME"
    echo "Executable is $FRAMEWORK_EXECUTABLE_PATH"

    EXTRACTED_ARCHS=()

    for ARCH in $ARCHS
    do
        echo "Extracting $ARCH from $FRAMEWORK_EXECUTABLE_NAME"
        lipo -extract "$ARCH" "$FRAMEWORK_EXECUTABLE_PATH" -o "$FRAMEWORK_EXECUTABLE_PATH-$ARCH"
        EXTRACTED_ARCHS+=("$FRAMEWORK_EXECUTABLE_PATH-$ARCH")
    done

    echo "Merging extracted architectures: ${ARCHS}"
    lipo -o "$FRAMEWORK_EXECUTABLE_PATH-merged" -create "${EXTRACTED_ARCHS[@]}"
    rm "${EXTRACTED_ARCHS[@]}"

    echo "Replacing original executable with thinned version"
    rm "$FRAMEWORK_EXECUTABLE_PATH"
    mv "$FRAMEWORK_EXECUTABLE_PATH-merged" "$FRAMEWORK_EXECUTABLE_PATH"

done
  1. IMPORTANT: When working on your app in XCode, delete this run script because it throws errors during regular debug development.

Agency IDs

The following ids are to be used in the various method calls above where "agency_id" is required. We are adding new markets reguarly.

Description Agency ID / CS ID
Chicago CTA cta
Chicago Metra METRA
Chicago PACE PACE
Boston MBTA MBTA
Pittsburgh PAAC PAAC
Minneapolis / St. Paul Area MINNEAPOLIS
Washington DC Area DC
Los Angeles County LA
Philadelphia - SEPTA Bus SEPTABUS
Philadelphia - SEPTA Rail SEPTARAIL
Seattle Area (King County Metro) SEATTLE
Miami-Dade County Metro MIAMI
Portland - TriMet TRIMET
Salt Lake City Area - UTA UTA
Utah - Cache Valley Transit District CVTD
NYC - MTA Subway MTA_SUBWAY
NYC - MTA Bus MTA_BUS
NYC - MTA Metro North MTA_METRO_NORTH
NYC - MTA Long Island Railroad MTA_LONG_ISLAND
New Jersey - NJ Transit Rail NJT_RAIL
New Jersey - NJ Transit Bus NJT_BUS
San Diego - Metropolitan Transit System MTS
San Francisco - SFMTA SFMTA
San Francisco - Bay Area Rapid Transit BART
San Francisco - AC Transit AC
Baltimore - Maryland Transit Administration MARYLAND
Las Vegas - RTC RTC
Tampa - Hillsborough Area Regional Transit HART
London - Transport for London TFL
Toronto - Toronto Transit Commission TTC