diff --git a/packages/firebase_admob/.gitignore b/packages/firebase_admob/.gitignore
new file mode 100644
index 000000000000..14c7d4c3f73e
--- /dev/null
+++ b/packages/firebase_admob/.gitignore
@@ -0,0 +1,9 @@
+.DS_Store
+.atom/
+.idea
+.packages
+.pub/
+build/
+ios/.generated/
+packages
+pubspec.lock
diff --git a/packages/firebase_admob/CHANGELOG.md b/packages/firebase_admob/CHANGELOG.md
new file mode 100644
index 000000000000..15b0faabffb9
--- /dev/null
+++ b/packages/firebase_admob/CHANGELOG.md
@@ -0,0 +1,3 @@
+## [0.0.1]
+
+* Initial Release: not ready for production use
diff --git a/packages/firebase_admob/LICENSE b/packages/firebase_admob/LICENSE
new file mode 100644
index 000000000000..c89293372cf3
--- /dev/null
+++ b/packages/firebase_admob/LICENSE
@@ -0,0 +1,27 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
diff --git a/packages/firebase_admob/README.md b/packages/firebase_admob/README.md
new file mode 100644
index 000000000000..08c0787b5a49
--- /dev/null
+++ b/packages/firebase_admob/README.md
@@ -0,0 +1,8 @@
+# firebase_admob
+A Flutter plugin based on the [Firebase AdMob API](https://firebase.google.com/docs/admob/).
+
+*Warning*: This plugin is still under development, some AdMob features are not available yet and testing has been limited. [Feedback](https://github.com/flutter/flutter/issues) and [Pull Requests](https://github.com/flutter/plugins/pulls) are welcome.
+
+See https://github.com/flutter/plugins/pull/171 for a brief overview of the current API.
+
+For Flutter plugins for other Firebase products, see [FlutterFire.md](https://github.com/flutter/plugins/blob/master/FlutterFire.md).
diff --git a/packages/firebase_admob/android/.gitignore b/packages/firebase_admob/android/.gitignore
new file mode 100644
index 000000000000..5c4ef82869b5
--- /dev/null
+++ b/packages/firebase_admob/android/.gitignore
@@ -0,0 +1,12 @@
+*.iml
+.gradle
+/local.properties
+/.idea/workspace.xml
+/.idea/libraries
+.DS_Store
+/build
+/captures
+
+/gradle
+/gradlew
+/gradlew.bat
diff --git a/packages/firebase_admob/android/build.gradle b/packages/firebase_admob/android/build.gradle
new file mode 100644
index 000000000000..f8296d1524d3
--- /dev/null
+++ b/packages/firebase_admob/android/build.gradle
@@ -0,0 +1,36 @@
+group 'io.flutter.plugins.firebaseadmob'
+version '1.0-SNAPSHOT'
+
+buildscript {
+ repositories {
+ jcenter()
+ }
+
+ dependencies {
+ classpath 'com.android.tools.build:gradle:2.3.0'
+ }
+}
+
+allprojects {
+ repositories {
+ jcenter()
+ }
+}
+
+apply plugin: 'com.android.library'
+
+android {
+ compileSdkVersion 25
+ buildToolsVersion '25.0.3'
+
+ defaultConfig {
+ testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+ }
+ lintOptions {
+ disable 'InvalidPackage'
+ }
+ dependencies {
+ compile 'com.google.firebase:firebase-core:11.0.+'
+ compile 'com.google.firebase:firebase-ads:11.0.+'
+ }
+}
diff --git a/packages/firebase_admob/android/gradle.properties b/packages/firebase_admob/android/gradle.properties
new file mode 100644
index 000000000000..8bd86f680510
--- /dev/null
+++ b/packages/firebase_admob/android/gradle.properties
@@ -0,0 +1 @@
+org.gradle.jvmargs=-Xmx1536M
diff --git a/packages/firebase_admob/android/settings.gradle b/packages/firebase_admob/android/settings.gradle
new file mode 100644
index 000000000000..c8378038e665
--- /dev/null
+++ b/packages/firebase_admob/android/settings.gradle
@@ -0,0 +1 @@
+rootProject.name = 'firebase_admob'
diff --git a/packages/firebase_admob/android/src/main/AndroidManifest.xml b/packages/firebase_admob/android/src/main/AndroidManifest.xml
new file mode 100644
index 000000000000..331c1c617d45
--- /dev/null
+++ b/packages/firebase_admob/android/src/main/AndroidManifest.xml
@@ -0,0 +1,9 @@
+
+
+
+
+
+
diff --git a/packages/firebase_admob/android/src/main/java/io/flutter/plugins/firebaseadmob/FirebaseAdMobPlugin.java b/packages/firebase_admob/android/src/main/java/io/flutter/plugins/firebaseadmob/FirebaseAdMobPlugin.java
new file mode 100644
index 000000000000..18816c3622a1
--- /dev/null
+++ b/packages/firebase_admob/android/src/main/java/io/flutter/plugins/firebaseadmob/FirebaseAdMobPlugin.java
@@ -0,0 +1,122 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+package io.flutter.plugins.firebaseadmob;
+
+import android.app.Activity;
+import android.widget.LinearLayout;
+import com.google.android.gms.ads.InterstitialAd;
+import com.google.android.gms.ads.MobileAds;
+import com.google.firebase.FirebaseApp;
+import io.flutter.plugin.common.MethodCall;
+import io.flutter.plugin.common.MethodChannel;
+import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
+import io.flutter.plugin.common.MethodChannel.Result;
+import io.flutter.plugin.common.PluginRegistry.Registrar;
+import java.util.Map;
+
+public class FirebaseAdMobPlugin implements MethodCallHandler {
+ private static final String TAG = "flutter";
+
+ private final Activity activity;
+ private final MethodChannel channel;
+
+ private LinearLayout banner;
+ InterstitialAd interstitial;
+
+ public static void registerWith(Registrar registrar) {
+ final MethodChannel channel =
+ new MethodChannel(registrar.messenger(), "plugins.flutter.io/firebase_admob");
+ channel.setMethodCallHandler(new FirebaseAdMobPlugin(registrar.activity(), channel));
+ }
+
+ private FirebaseAdMobPlugin(Activity activity, MethodChannel channel) {
+ this.activity = activity;
+ this.channel = channel;
+ FirebaseApp.initializeApp(activity);
+ }
+
+ private void callInitialize(MethodCall call, Result result) {
+ String appId = call.argument("appId");
+ if (appId == null || appId.isEmpty()) {
+ result.error("no_app_id", "a non-empty AdMob appId was not provided", null);
+ return;
+ }
+ MobileAds.initialize(activity, appId);
+ result.success(Boolean.TRUE);
+ }
+
+ private void callLoadAd(MobileAd ad, MethodCall call, Result result) {
+ if (ad.status != MobileAd.Status.CREATED) {
+ if (ad.status == MobileAd.Status.FAILED)
+ result.error("load_failed_ad", "cannot reload a failed ad, id=" + ad.id, null);
+ else result.success(Boolean.TRUE); // The ad was already loaded.
+ return;
+ }
+
+ String unitId = call.argument("unitId");
+ if (unitId == null || unitId.isEmpty()) {
+ result.error("no_unit_id", "a non-empty unitId was not provided for ad id=" + ad.id, null);
+ return;
+ }
+ Map targetingInfo = call.argument("targetingInfo");
+ ad.load(unitId, targetingInfo);
+ result.success(Boolean.TRUE);
+ }
+
+ private void callShowAd(int id, MethodCall call, Result result) {
+ MobileAd ad = MobileAd.getAdForId(id);
+ if (ad == null) {
+ result.error("ad_not_loaded", "show failed, the specified ad was not loaded id=" + id, null);
+ return;
+ }
+ ad.show();
+ result.success(Boolean.TRUE);
+ }
+
+ private void callDisposeAd(int id, MethodCall call, Result result) {
+ MobileAd ad = MobileAd.getAdForId(id);
+ if (ad == null) {
+ result.error("no_ad_for_id", "dispose failed, no add exists for id=" + id, null);
+ return;
+ }
+
+ ad.dispose();
+ result.success(Boolean.TRUE);
+ }
+
+ @Override
+ public void onMethodCall(MethodCall call, Result result) {
+ if (call.method.equals("initialize")) {
+ callInitialize(call, result);
+ return;
+ }
+
+ Integer id = call.argument("id");
+ if (id == null) {
+ result.error(
+ "no_id",
+ "all FirebaseAdMobPlugin method calls must specify an integer mobile ad id",
+ null);
+ return;
+ }
+
+ switch (call.method) {
+ case "loadBannerAd":
+ callLoadAd(MobileAd.createBanner(id, activity, channel), call, result);
+ break;
+ case "loadInterstitialAd":
+ callLoadAd(MobileAd.createInterstitial(id, activity, channel), call, result);
+ break;
+ case "showAd":
+ callShowAd(id, call, result);
+ break;
+ case "disposeAd":
+ callDisposeAd(id, call, result);
+ break;
+ default:
+ result.notImplemented();
+ }
+ }
+}
diff --git a/packages/firebase_admob/android/src/main/java/io/flutter/plugins/firebaseadmob/MobileAd.java b/packages/firebase_admob/android/src/main/java/io/flutter/plugins/firebaseadmob/MobileAd.java
new file mode 100644
index 000000000000..cd99a45ce340
--- /dev/null
+++ b/packages/firebase_admob/android/src/main/java/io/flutter/plugins/firebaseadmob/MobileAd.java
@@ -0,0 +1,307 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+package io.flutter.plugins.firebaseadmob;
+
+import android.app.Activity;
+import android.util.Log;
+import android.util.SparseArray;
+import android.view.Gravity;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.LinearLayout;
+import com.google.android.gms.ads.AdListener;
+import com.google.android.gms.ads.AdRequest;
+import com.google.android.gms.ads.AdSize;
+import com.google.android.gms.ads.AdView;
+import com.google.android.gms.ads.InterstitialAd;
+import io.flutter.plugin.common.MethodChannel;
+import java.util.ArrayList;
+import java.util.Date;
+import java.util.HashMap;
+import java.util.Map;
+
+abstract class MobileAd extends AdListener {
+ private static final String TAG = "flutter";
+ private static SparseArray allAds = new SparseArray();
+
+ final Activity activity;
+ final MethodChannel channel;
+ final int id;
+ Status status;
+
+ enum Status {
+ CREATED,
+ LOADING,
+ FAILED,
+ PENDING, // The ad will be shown when status is changed to LOADED.
+ LOADED,
+ }
+
+ private MobileAd(int id, Activity activity, MethodChannel channel) {
+ this.id = id;
+ this.activity = activity;
+ this.channel = channel;
+ this.status = Status.CREATED;
+ allAds.put(id, this);
+ }
+
+ static Banner createBanner(Integer id, Activity activity, MethodChannel channel) {
+ MobileAd ad = getAdForId(id);
+ return (ad != null) ? (Banner) ad : new Banner(id, activity, channel);
+ }
+
+ static Interstitial createInterstitial(Integer id, Activity activity, MethodChannel channel) {
+ MobileAd ad = getAdForId(id);
+ return (ad != null) ? (Interstitial) ad : new Interstitial(id, activity, channel);
+ }
+
+ static MobileAd getAdForId(Integer id) {
+ return allAds.get(id);
+ }
+
+ Status getStatus() {
+ return status;
+ }
+
+ abstract void load(String unitId, Map targetingInfo);
+
+ abstract void show();
+
+ void dispose() {
+ allAds.remove(id);
+ }
+
+ private String getTargetingInfoString(String key, Object value) {
+ if (value == null) return null;
+ if (!(value instanceof String)) {
+ Log.w(TAG, "targeting info " + key + ": expected a String, mobileAdId=" + id);
+ return null;
+ }
+ String stringValue = (String) value;
+ if (stringValue.isEmpty()) {
+ Log.w(TAG, "targeting info " + key + ": expected a non-empty String, mobileAdId=" + id);
+ return null;
+ }
+ return stringValue;
+ }
+
+ private Boolean getTargetingInfoBoolean(String key, Object value) {
+ if (value == null) return null;
+ if (!(value instanceof Boolean)) {
+ Log.w(TAG, "targeting info " + key + ": expected a boolean, mobileAdId=" + id);
+ return null;
+ }
+ return (Boolean) value;
+ }
+
+ private Integer getTargetingInfoInteger(String key, Object value) {
+ if (value == null) return null;
+ if (!(value instanceof Integer)) {
+ Log.w(TAG, "targeting info " + key + ": expected an integer, mobileAdId=" + id);
+ return null;
+ }
+ return (Integer) value;
+ }
+
+ private ArrayList getTargetingInfoArrayList(String key, Object value) {
+ if (value == null) return null;
+ if (!(value instanceof ArrayList)) {
+ Log.w(TAG, "targeting info " + key + ": expected an ArrayList, mobileAdId=" + id);
+ return null;
+ }
+ return (ArrayList) value;
+ }
+
+ AdRequest.Builder createAdRequestBuilder(Map info) {
+ AdRequest.Builder builder = new AdRequest.Builder();
+ if (info == null) return builder;
+
+ ArrayList testDevices = getTargetingInfoArrayList("testDevices", info.get("testDevices"));
+ if (testDevices != null) {
+ for (Object deviceValue : testDevices) {
+ String device = getTargetingInfoString("testDevices element", deviceValue);
+ if (device != null) builder.addTestDevice(device);
+ }
+ }
+
+ ArrayList keywords = getTargetingInfoArrayList("keywords", info.get("keywords"));
+ if (keywords != null) {
+ for (Object keywordValue : keywords) {
+ String keyword = getTargetingInfoString("keywords element", keywordValue);
+ if (keyword != null) builder.addKeyword(keyword);
+ }
+ }
+
+ String contentUrl = getTargetingInfoString("contentUrl", info.get("contentUrl"));
+ if (contentUrl != null) builder.setContentUrl(contentUrl);
+
+ Object birthday = info.get("birthday");
+ if (birthday != null) {
+ if (!(birthday instanceof Long))
+ Log.w(TAG, "targeting info birthday: expected a long integer, mobileAdId=" + id);
+ else builder.setBirthday(new Date((Long) birthday));
+ }
+
+ Integer gender = getTargetingInfoInteger("gender", info.get("gender"));
+ if (gender != null) {
+ switch (gender.intValue()) {
+ case 0: // MobileAdGender.unknown
+ case 1: // MobileAdGender.male
+ case 2: // MobileAdGender.female
+ builder.setGender(gender.intValue());
+ break;
+ default:
+ Log.w(TAG, "targeting info gender: invalid value, mobileAdId=" + id);
+ }
+ }
+
+ Boolean designedForFamilies =
+ getTargetingInfoBoolean("designedForFamilies", info.get("designedForFamilies"));
+ if (designedForFamilies != null) builder.setIsDesignedForFamilies(designedForFamilies);
+
+ Boolean childDirected = getTargetingInfoBoolean("childDirected", info.get("childDirected"));
+ if (childDirected != null) builder.tagForChildDirectedTreatment(childDirected);
+
+ String requestAgent = getTargetingInfoString("requestAgent", info.get("requestAgent"));
+ if (requestAgent != null) builder.setRequestAgent(requestAgent);
+
+ return builder;
+ }
+
+ private Map argumentsMap(Object... args) {
+ Map arguments = new HashMap();
+ arguments.put("id", id);
+ for (int i = 0; i < args.length; i += 2) arguments.put(args[i].toString(), args[i + 1]);
+ return arguments;
+ }
+
+ @Override
+ public void onAdLoaded() {
+ boolean statusWasPending = status == Status.PENDING;
+ status = Status.LOADED;
+ channel.invokeMethod("onAdLoaded", argumentsMap());
+ if (statusWasPending) show();
+ }
+
+ @Override
+ public void onAdFailedToLoad(int errorCode) {
+ Log.w(TAG, "onAdFailedToLoad: " + errorCode);
+ status = Status.FAILED;
+ channel.invokeMethod("onAdFailedToLoad", argumentsMap("errorCode", errorCode));
+ }
+
+ @Override
+ public void onAdOpened() {
+ channel.invokeMethod("onAdOpened", argumentsMap());
+ }
+
+ @Override
+ public void onAdClicked() {
+ channel.invokeMethod("onAdClicked", argumentsMap());
+ }
+
+ @Override
+ public void onAdImpression() {
+ channel.invokeMethod("onAdImpression", argumentsMap());
+ }
+
+ @Override
+ public void onAdLeftApplication() {
+ channel.invokeMethod("onAdLeftApplication", argumentsMap());
+ }
+
+ @Override
+ public void onAdClosed() {
+ channel.invokeMethod("onAdClosed", argumentsMap());
+ }
+
+ static class Banner extends MobileAd {
+ private AdView adView;
+
+ private Banner(Integer id, Activity activity, MethodChannel channel) {
+ super(id, activity, channel);
+ }
+
+ @Override
+ void load(String unitId, Map targetingInfo) {
+ if (status != Status.CREATED) return;
+ status = Status.LOADING;
+
+ adView = new AdView(activity);
+ adView.setAdSize(AdSize.SMART_BANNER);
+ adView.setAdUnitId(unitId);
+ adView.setAdListener(this);
+
+ AdRequest.Builder adRequestBuilder = createAdRequestBuilder(targetingInfo);
+ adView.loadAd(adRequestBuilder.build());
+ }
+
+ @Override
+ void show() {
+ if (status == Status.LOADING) {
+ status = Status.PENDING;
+ return;
+ }
+ if (status != Status.LOADED) return;
+
+ if (activity.findViewById(id) == null) {
+ LinearLayout content = new LinearLayout(activity);
+ content.setId(id);
+ content.setOrientation(LinearLayout.VERTICAL);
+ content.setGravity(Gravity.BOTTOM);
+ content.addView(adView);
+
+ activity.addContentView(
+ content,
+ new ViewGroup.LayoutParams(
+ ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.MATCH_PARENT));
+ }
+ }
+
+ @Override
+ void dispose() {
+ super.dispose();
+
+ View contentView = activity.findViewById(id);
+ if (contentView == null || !(contentView.getParent() instanceof ViewGroup)) return;
+
+ adView.destroy();
+
+ ViewGroup contentParent = (ViewGroup) (contentView.getParent());
+ contentParent.removeView(contentView);
+ }
+ }
+
+ static class Interstitial extends MobileAd {
+ private InterstitialAd interstitial = null;
+
+ private Interstitial(int id, Activity activity, MethodChannel channel) {
+ super(id, activity, channel);
+ }
+
+ @Override
+ void load(String unitId, Map targetingInfo) {
+ status = Status.LOADING;
+
+ interstitial = new InterstitialAd(activity);
+ interstitial.setAdUnitId(unitId);
+
+ AdRequest.Builder adRequestBuilder = createAdRequestBuilder(targetingInfo);
+ interstitial.setAdListener(this);
+ interstitial.loadAd(adRequestBuilder.build());
+ }
+
+ @Override
+ void show() {
+ if (status == Status.LOADING) {
+ status = Status.PENDING;
+ return;
+ }
+ interstitial.show();
+ }
+
+ // It is not possible to hide/remove/destroy an AdMob interstitial Ad.
+ }
+}
diff --git a/packages/firebase_admob/example/.gitignore b/packages/firebase_admob/example/.gitignore
new file mode 100644
index 000000000000..eb15c3d27cab
--- /dev/null
+++ b/packages/firebase_admob/example/.gitignore
@@ -0,0 +1,10 @@
+.DS_Store
+.atom/
+.idea
+.packages
+.pub/
+build/
+ios/.generated/
+packages
+pubspec.lock
+.flutter-plugins
diff --git a/packages/firebase_admob/example/README.md b/packages/firebase_admob/example/README.md
new file mode 100644
index 000000000000..148a11720dd8
--- /dev/null
+++ b/packages/firebase_admob/example/README.md
@@ -0,0 +1,8 @@
+# firebase_admob_example
+
+Demonstrates how to use the firebase_admob plugin.
+
+## Getting Started
+
+For help getting started with Flutter, view our online
+[documentation](http://flutter.io/).
diff --git a/packages/firebase_admob/example/android.iml b/packages/firebase_admob/example/android.iml
new file mode 100644
index 000000000000..462b903e05b6
--- /dev/null
+++ b/packages/firebase_admob/example/android.iml
@@ -0,0 +1,12 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/example/android/.gitignore b/packages/firebase_admob/example/android/.gitignore
new file mode 100644
index 000000000000..1fd9325cac44
--- /dev/null
+++ b/packages/firebase_admob/example/android/.gitignore
@@ -0,0 +1,13 @@
+*.iml
+.gradle
+/local.properties
+/.idea/workspace.xml
+/.idea/libraries
+.DS_Store
+/build
+/captures
+GeneratedPluginRegistrant.java
+
+/gradle
+/gradlew
+/gradlew.bat
diff --git a/packages/firebase_admob/example/android/app/build.gradle b/packages/firebase_admob/example/android/app/build.gradle
new file mode 100644
index 000000000000..0403a2fd6096
--- /dev/null
+++ b/packages/firebase_admob/example/android/app/build.gradle
@@ -0,0 +1,49 @@
+def localProperties = new Properties()
+def localPropertiesFile = rootProject.file('local.properties')
+if (localPropertiesFile.exists()) {
+ localPropertiesFile.withInputStream { stream ->
+ localProperties.load(stream)
+ }
+}
+
+def flutterRoot = localProperties.getProperty('flutter.sdk')
+if (flutterRoot == null) {
+ throw new GradleException("Flutter SDK not found. Define location with flutter.sdk in the local.properties file.")
+}
+
+apply plugin: 'com.android.application'
+apply from: "$flutterRoot/packages/flutter_tools/gradle/flutter.gradle"
+
+android {
+ compileSdkVersion 25
+ buildToolsVersion '25.0.3'
+
+ lintOptions {
+ disable 'InvalidPackage'
+ }
+
+ defaultConfig {
+ testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+ applicationId "io.flutter.plugins.firebaseadmobexample"
+ }
+
+ buildTypes {
+ release {
+ // TODO: Add your own signing config for the release build.
+ // Signing with the debug keys for now, so `flutter run --release` works.
+ signingConfig signingConfigs.debug
+ }
+ }
+}
+
+flutter {
+ source '../..'
+}
+
+dependencies {
+ androidTestCompile 'com.android.support:support-annotations:25.2.0'
+ androidTestCompile 'com.android.support.test:runner:0.5'
+ androidTestCompile 'com.android.support.test:rules:0.5'
+}
+
+apply plugin: 'com.google.gms.google-services'
diff --git a/packages/firebase_admob/example/android/app/google-services.json b/packages/firebase_admob/example/android/app/google-services.json
new file mode 100644
index 000000000000..ee45722eb3f3
--- /dev/null
+++ b/packages/firebase_admob/example/android/app/google-services.json
@@ -0,0 +1,99 @@
+{
+ "project_info": {
+ "project_number": "395789065833",
+ "firebase_url": "https://admob-test-66963.firebaseio.com",
+ "project_id": "admob-test-66963",
+ "storage_bucket": "admob-test-66963.appspot.com"
+ },
+ "client": [
+ {
+ "client_info": {
+ "mobilesdk_app_id": "1:395789065833:android:cf455c18665bfb73",
+ "android_client_info": {
+ "package_name": "com.yourcompany.ads"
+ }
+ },
+ "oauth_client": [
+ {
+ "client_id": "395789065833-72h13sj0ba0alalnijd26ni8q19d7s27.apps.googleusercontent.com",
+ "client_type": 1,
+ "android_info": {
+ "package_name": "com.yourcompany.ads",
+ "certificate_hash": "266da292ae1273cd3bedde8f7e5d82080234d46f"
+ }
+ },
+ {
+ "client_id": "395789065833-soebkshopimb9sbsugdqeohjcrqc1h5v.apps.googleusercontent.com",
+ "client_type": 3
+ }
+ ],
+ "api_key": [
+ {
+ "current_key": "AIzaSyAckYXawYez8b74iXfGVM7OBowoqjhiZ38"
+ }
+ ],
+ "services": {
+ "analytics_service": {
+ "status": 1
+ },
+ "appinvite_service": {
+ "status": 2,
+ "other_platform_oauth_client": [
+ {
+ "client_id": "395789065833-soebkshopimb9sbsugdqeohjcrqc1h5v.apps.googleusercontent.com",
+ "client_type": 3
+ }
+ ]
+ },
+ "ads_service": {
+ "status": 2
+ }
+ }
+ },
+ {
+ "client_info": {
+ "mobilesdk_app_id": "1:395789065833:android:d848e43bc35ca200",
+ "android_client_info": {
+ "package_name": "io.flutter.plugins.firebaseadmobexample"
+ }
+ },
+ "oauth_client": [
+ {
+ "client_id": "395789065833-jo1lkael0e4be0r2o2tan6qm3khfr6rs.apps.googleusercontent.com",
+ "client_type": 1,
+ "android_info": {
+ "package_name": "io.flutter.plugins.firebaseadmobexample",
+ "certificate_hash": "266da292ae1273cd3bedde8f7e5d82080234d46f"
+ }
+ },
+ {
+ "client_id": "395789065833-soebkshopimb9sbsugdqeohjcrqc1h5v.apps.googleusercontent.com",
+ "client_type": 3
+ }
+ ],
+ "api_key": [
+ {
+ "current_key": "AIzaSyAckYXawYez8b74iXfGVM7OBowoqjhiZ38"
+ }
+ ],
+ "services": {
+ "analytics_service": {
+ "status": 1
+ },
+ "appinvite_service": {
+ "status": 2,
+ "other_platform_oauth_client": [
+ {
+ "client_id": "395789065833-soebkshopimb9sbsugdqeohjcrqc1h5v.apps.googleusercontent.com",
+ "client_type": 3
+ }
+ ]
+ },
+ "ads_service": {
+ "status": 2
+ }
+ }
+ }
+ ],
+ "configuration_version": "1"
+}
\ No newline at end of file
diff --git a/packages/firebase_admob/example/android/app/src/main/AndroidManifest.xml b/packages/firebase_admob/example/android/app/src/main/AndroidManifest.xml
new file mode 100644
index 000000000000..b8a34e7b4823
--- /dev/null
+++ b/packages/firebase_admob/example/android/app/src/main/AndroidManifest.xml
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/example/android/app/src/main/java/io/flutter/plugins/firebaseadmobexample/MainActivity.java b/packages/firebase_admob/example/android/app/src/main/java/io/flutter/plugins/firebaseadmobexample/MainActivity.java
new file mode 100644
index 000000000000..14775bbf2fb7
--- /dev/null
+++ b/packages/firebase_admob/example/android/app/src/main/java/io/flutter/plugins/firebaseadmobexample/MainActivity.java
@@ -0,0 +1,17 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+package io.flutter.plugins.firebaseadmobexample;
+
+import android.os.Bundle;
+import io.flutter.app.FlutterActivity;
+import io.flutter.plugins.GeneratedPluginRegistrant;
+
+public class MainActivity extends FlutterActivity {
+ @Override
+ protected void onCreate(Bundle savedInstanceState) {
+ super.onCreate(savedInstanceState);
+ GeneratedPluginRegistrant.registerWith(this);
+ }
+}
diff --git a/packages/firebase_admob/example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png b/packages/firebase_admob/example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
new file mode 100644
index 000000000000..db77bb4b7b09
Binary files /dev/null and b/packages/firebase_admob/example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png differ
diff --git a/packages/firebase_admob/example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png b/packages/firebase_admob/example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png
new file mode 100644
index 000000000000..17987b79bb8a
Binary files /dev/null and b/packages/firebase_admob/example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png differ
diff --git a/packages/firebase_admob/example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/packages/firebase_admob/example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
new file mode 100644
index 000000000000..09d4391482be
Binary files /dev/null and b/packages/firebase_admob/example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png differ
diff --git a/packages/firebase_admob/example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/packages/firebase_admob/example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
new file mode 100644
index 000000000000..d5f1c8d34e7a
Binary files /dev/null and b/packages/firebase_admob/example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png differ
diff --git a/packages/firebase_admob/example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/packages/firebase_admob/example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
new file mode 100644
index 000000000000..4d6372eebdb2
Binary files /dev/null and b/packages/firebase_admob/example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png differ
diff --git a/packages/firebase_admob/example/android/build.gradle b/packages/firebase_admob/example/android/build.gradle
new file mode 100644
index 000000000000..2023b0f5725e
--- /dev/null
+++ b/packages/firebase_admob/example/android/build.gradle
@@ -0,0 +1,30 @@
+buildscript {
+ repositories {
+ jcenter()
+ }
+
+ dependencies {
+ classpath 'com.android.tools.build:gradle:2.2.3'
+ classpath 'com.google.gms:google-services:3.0.0'
+ }
+}
+
+allprojects {
+ repositories {
+ jcenter()
+ }
+}
+
+rootProject.buildDir = '../build'
+subprojects {
+ project.buildDir = "${rootProject.buildDir}/${project.name}"
+ project.evaluationDependsOn(':app')
+}
+
+task clean(type: Delete) {
+ delete rootProject.buildDir
+}
+
+task wrapper(type: Wrapper) {
+ gradleVersion = '2.14.1'
+}
diff --git a/packages/firebase_admob/example/android/gradle.properties b/packages/firebase_admob/example/android/gradle.properties
new file mode 100644
index 000000000000..8bd86f680510
--- /dev/null
+++ b/packages/firebase_admob/example/android/gradle.properties
@@ -0,0 +1 @@
+org.gradle.jvmargs=-Xmx1536M
diff --git a/packages/firebase_admob/example/android/settings.gradle b/packages/firebase_admob/example/android/settings.gradle
new file mode 100644
index 000000000000..115da6cb4f4d
--- /dev/null
+++ b/packages/firebase_admob/example/android/settings.gradle
@@ -0,0 +1,15 @@
+include ':app'
+
+def flutterProjectRoot = rootProject.projectDir.parentFile.toPath()
+
+def plugins = new Properties()
+def pluginsFile = new File(flutterProjectRoot.toFile(), '.flutter-plugins')
+if (pluginsFile.exists()) {
+ pluginsFile.withInputStream { stream -> plugins.load(stream) }
+}
+
+plugins.each { name, path ->
+ def pluginDirectory = flutterProjectRoot.resolve(path).resolve('android').toFile()
+ include ":$name"
+ project(":$name").projectDir = pluginDirectory
+}
diff --git a/packages/firebase_admob/example/firebase_admob_example.iml b/packages/firebase_admob/example/firebase_admob_example.iml
new file mode 100644
index 000000000000..9d5dae19540c
--- /dev/null
+++ b/packages/firebase_admob/example/firebase_admob_example.iml
@@ -0,0 +1,15 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/packages/firebase_admob/example/firebase_admob_example_android.iml b/packages/firebase_admob/example/firebase_admob_example_android.iml
new file mode 100644
index 000000000000..462b903e05b6
--- /dev/null
+++ b/packages/firebase_admob/example/firebase_admob_example_android.iml
@@ -0,0 +1,12 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/example/ios/.gitignore b/packages/firebase_admob/example/ios/.gitignore
new file mode 100644
index 000000000000..38864eed23ef
--- /dev/null
+++ b/packages/firebase_admob/example/ios/.gitignore
@@ -0,0 +1,41 @@
+.idea/
+.vagrant/
+.sconsign.dblite
+.svn/
+
+.DS_Store
+*.swp
+profile
+
+DerivedData/
+build/
+GeneratedPluginRegistrant.h
+GeneratedPluginRegistrant.m
+
+*.pbxuser
+*.mode1v3
+*.mode2v3
+*.perspectivev3
+
+!default.pbxuser
+!default.mode1v3
+!default.mode2v3
+!default.perspectivev3
+
+xcuserdata
+
+*.moved-aside
+
+*.pyc
+*sync/
+Icon?
+.tags*
+
+/Flutter/app.flx
+/Flutter/app.zip
+/Flutter/App.framework
+/Flutter/Flutter.framework
+/Flutter/Generated.xcconfig
+/ServiceDefinitions.json
+
+Pods/
diff --git a/packages/firebase_admob/example/ios/Flutter/AppFrameworkInfo.plist b/packages/firebase_admob/example/ios/Flutter/AppFrameworkInfo.plist
new file mode 100644
index 000000000000..6c2de8086bcd
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Flutter/AppFrameworkInfo.plist
@@ -0,0 +1,30 @@
+
+
+
+
+ CFBundleDevelopmentRegion
+ en
+ CFBundleExecutable
+ App
+ CFBundleIdentifier
+ io.flutter.flutter.app
+ CFBundleInfoDictionaryVersion
+ 6.0
+ CFBundleName
+ App
+ CFBundlePackageType
+ FMWK
+ CFBundleShortVersionString
+ 1.0
+ CFBundleSignature
+ ????
+ CFBundleVersion
+ 1.0
+ UIRequiredDeviceCapabilities
+
+ arm64
+
+ MinimumOSVersion
+ 8.0
+
+
diff --git a/packages/firebase_admob/example/ios/Flutter/Debug.xcconfig b/packages/firebase_admob/example/ios/Flutter/Debug.xcconfig
new file mode 100644
index 000000000000..592ceee85b89
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Flutter/Debug.xcconfig
@@ -0,0 +1 @@
+#include "Generated.xcconfig"
diff --git a/packages/firebase_admob/example/ios/Flutter/Release.xcconfig b/packages/firebase_admob/example/ios/Flutter/Release.xcconfig
new file mode 100644
index 000000000000..592ceee85b89
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Flutter/Release.xcconfig
@@ -0,0 +1 @@
+#include "Generated.xcconfig"
diff --git a/packages/firebase_admob/example/ios/Podfile b/packages/firebase_admob/example/ios/Podfile
new file mode 100644
index 000000000000..90b5f651fb63
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Podfile
@@ -0,0 +1,36 @@
+# Uncomment this line to define a global platform for your project
+# platform :ios, '9.0'
+
+if ENV['FLUTTER_FRAMEWORK_DIR'] == nil
+ abort('Please set FLUTTER_FRAMEWORK_DIR to the directory containing Flutter.framework')
+end
+
+target 'Runner' do
+ # Pods for Runner
+
+ # Flutter Pods
+ pod 'Flutter', :path => ENV['FLUTTER_FRAMEWORK_DIR']
+
+ if File.exists? '../.flutter-plugins'
+ flutter_root = File.expand_path('..')
+ File.foreach('../.flutter-plugins') { |line|
+ plugin = line.split(pattern='=')
+ if plugin.length == 2
+ name = plugin[0].strip()
+ path = plugin[1].strip()
+ resolved_path = File.expand_path("#{path}/ios", flutter_root)
+ pod name, :path => resolved_path
+ else
+ puts "Invalid plugin specification: #{line}"
+ end
+ }
+ end
+end
+
+post_install do |installer|
+ installer.pods_project.targets.each do |target|
+ target.build_configurations.each do |config|
+ config.build_settings['ENABLE_BITCODE'] = 'NO'
+ end
+ end
+end
diff --git a/packages/firebase_admob/example/ios/Runner.xcodeproj/project.pbxproj b/packages/firebase_admob/example/ios/Runner.xcodeproj/project.pbxproj
new file mode 100644
index 000000000000..a63f59996fd5
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner.xcodeproj/project.pbxproj
@@ -0,0 +1,498 @@
+// !$*UTF8*$!
+{
+ archiveVersion = 1;
+ classes = {
+ };
+ objectVersion = 46;
+ objects = {
+
+/* Begin PBXBuildFile section */
+ 1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */ = {isa = PBXBuildFile; fileRef = 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */; };
+ 3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; };
+ 3B80C3941E831B6300D905FE /* App.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3B80C3931E831B6300D905FE /* App.framework */; };
+ 3B80C3951E831B6300D905FE /* App.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 3B80C3931E831B6300D905FE /* App.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; };
+ 6E27B1151F0DAFA70028FD65 /* GoogleService-Info.plist in Resources */ = {isa = PBXBuildFile; fileRef = 6E27B1141F0DAFA70028FD65 /* GoogleService-Info.plist */; };
+ 9705A1C61CF904A100538489 /* Flutter.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 9740EEBA1CF902C7004384FC /* Flutter.framework */; };
+ 9705A1C71CF904A300538489 /* Flutter.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 9740EEBA1CF902C7004384FC /* Flutter.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; };
+ 9740EEB41CF90195004384FC /* Debug.xcconfig in Resources */ = {isa = PBXBuildFile; fileRef = 9740EEB21CF90195004384FC /* Debug.xcconfig */; };
+ 9740EEB51CF90195004384FC /* Generated.xcconfig in Resources */ = {isa = PBXBuildFile; fileRef = 9740EEB31CF90195004384FC /* Generated.xcconfig */; };
+ 9740EEBB1CF902C7004384FC /* app.flx in Resources */ = {isa = PBXBuildFile; fileRef = 9740EEB71CF902C7004384FC /* app.flx */; };
+ 978B8F6F1D3862AE00F588F7 /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 7AFFD8EE1D35381100E5BB4D /* AppDelegate.m */; };
+ 97C146F31CF9000F007C117D /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 97C146F21CF9000F007C117D /* main.m */; };
+ 97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; };
+ 97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; };
+ 97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; };
+ FBE669D215209F1F44CEEB21 /* libPods-Runner.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 83F369F228D3A43519CEE308 /* libPods-Runner.a */; };
+/* End PBXBuildFile section */
+
+/* Begin PBXCopyFilesBuildPhase section */
+ 9705A1C41CF9048500538489 /* Embed Frameworks */ = {
+ isa = PBXCopyFilesBuildPhase;
+ buildActionMask = 2147483647;
+ dstPath = "";
+ dstSubfolderSpec = 10;
+ files = (
+ 3B80C3951E831B6300D905FE /* App.framework in Embed Frameworks */,
+ 9705A1C71CF904A300538489 /* Flutter.framework in Embed Frameworks */,
+ );
+ name = "Embed Frameworks";
+ runOnlyForDeploymentPostprocessing = 0;
+ };
+/* End PBXCopyFilesBuildPhase section */
+
+/* Begin PBXFileReference section */
+ 1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = ""; };
+ 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = ""; };
+ 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = ""; };
+ 3B80C3931E831B6300D905FE /* App.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = App.framework; path = Flutter/App.framework; sourceTree = ""; };
+ 6E27B1141F0DAFA70028FD65 /* GoogleService-Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; path = "GoogleService-Info.plist"; sourceTree = ""; };
+ 7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = ""; };
+ 7AFFD8ED1D35381100E5BB4D /* AppDelegate.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = AppDelegate.h; sourceTree = ""; };
+ 7AFFD8EE1D35381100E5BB4D /* AppDelegate.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = AppDelegate.m; sourceTree = ""; };
+ 83F369F228D3A43519CEE308 /* libPods-Runner.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-Runner.a"; sourceTree = BUILT_PRODUCTS_DIR; };
+ 9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = ""; };
+ 9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = ""; };
+ 9740EEB71CF902C7004384FC /* app.flx */ = {isa = PBXFileReference; lastKnownFileType = file; name = app.flx; path = Flutter/app.flx; sourceTree = ""; };
+ 9740EEBA1CF902C7004384FC /* Flutter.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Flutter.framework; path = Flutter/Flutter.framework; sourceTree = ""; };
+ 97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; };
+ 97C146F21CF9000F007C117D /* main.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = main.m; sourceTree = ""; };
+ 97C146FB1CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = ""; };
+ 97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; };
+ 97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; };
+ 97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; };
+/* End PBXFileReference section */
+
+/* Begin PBXFrameworksBuildPhase section */
+ 97C146EB1CF9000F007C117D /* Frameworks */ = {
+ isa = PBXFrameworksBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ 9705A1C61CF904A100538489 /* Flutter.framework in Frameworks */,
+ 3B80C3941E831B6300D905FE /* App.framework in Frameworks */,
+ FBE669D215209F1F44CEEB21 /* libPods-Runner.a in Frameworks */,
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ };
+/* End PBXFrameworksBuildPhase section */
+
+/* Begin PBXGroup section */
+ 07A52D07C2C05D9527204891 /* Pods */ = {
+ isa = PBXGroup;
+ children = (
+ );
+ name = Pods;
+ sourceTree = "";
+ };
+ 5B6AAA352BAC85BF7DD68C46 /* Frameworks */ = {
+ isa = PBXGroup;
+ children = (
+ 83F369F228D3A43519CEE308 /* libPods-Runner.a */,
+ );
+ name = Frameworks;
+ sourceTree = "";
+ };
+ 9740EEB11CF90186004384FC /* Flutter */ = {
+ isa = PBXGroup;
+ children = (
+ 9740EEB71CF902C7004384FC /* app.flx */,
+ 3B80C3931E831B6300D905FE /* App.framework */,
+ 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */,
+ 9740EEBA1CF902C7004384FC /* Flutter.framework */,
+ 9740EEB21CF90195004384FC /* Debug.xcconfig */,
+ 7AFA3C8E1D35360C0083082E /* Release.xcconfig */,
+ 9740EEB31CF90195004384FC /* Generated.xcconfig */,
+ );
+ name = Flutter;
+ sourceTree = "";
+ };
+ 97C146E51CF9000F007C117D = {
+ isa = PBXGroup;
+ children = (
+ 9740EEB11CF90186004384FC /* Flutter */,
+ 97C146F01CF9000F007C117D /* Runner */,
+ 97C146EF1CF9000F007C117D /* Products */,
+ 07A52D07C2C05D9527204891 /* Pods */,
+ 5B6AAA352BAC85BF7DD68C46 /* Frameworks */,
+ );
+ sourceTree = "";
+ };
+ 97C146EF1CF9000F007C117D /* Products */ = {
+ isa = PBXGroup;
+ children = (
+ 97C146EE1CF9000F007C117D /* Runner.app */,
+ );
+ name = Products;
+ sourceTree = "";
+ };
+ 97C146F01CF9000F007C117D /* Runner */ = {
+ isa = PBXGroup;
+ children = (
+ 6E27B1141F0DAFA70028FD65 /* GoogleService-Info.plist */,
+ 7AFFD8ED1D35381100E5BB4D /* AppDelegate.h */,
+ 7AFFD8EE1D35381100E5BB4D /* AppDelegate.m */,
+ 97C146FA1CF9000F007C117D /* Main.storyboard */,
+ 97C146FD1CF9000F007C117D /* Assets.xcassets */,
+ 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */,
+ 97C147021CF9000F007C117D /* Info.plist */,
+ 97C146F11CF9000F007C117D /* Supporting Files */,
+ 1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */,
+ 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */,
+ );
+ path = Runner;
+ sourceTree = "";
+ };
+ 97C146F11CF9000F007C117D /* Supporting Files */ = {
+ isa = PBXGroup;
+ children = (
+ 97C146F21CF9000F007C117D /* main.m */,
+ );
+ name = "Supporting Files";
+ sourceTree = "";
+ };
+/* End PBXGroup section */
+
+/* Begin PBXNativeTarget section */
+ 97C146ED1CF9000F007C117D /* Runner */ = {
+ isa = PBXNativeTarget;
+ buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */;
+ buildPhases = (
+ 2C1A4E9A6849F7CC6B679EA5 /* [CP] Check Pods Manifest.lock */,
+ 9740EEB61CF901F6004384FC /* Run Script */,
+ 97C146EA1CF9000F007C117D /* Sources */,
+ 97C146EB1CF9000F007C117D /* Frameworks */,
+ 97C146EC1CF9000F007C117D /* Resources */,
+ 9705A1C41CF9048500538489 /* Embed Frameworks */,
+ 3B06AD1E1E4923F5004D2608 /* Thin Binary */,
+ E42D8ABA8B028F5FFACB19B6 /* [CP] Embed Pods Frameworks */,
+ 14942D2CEC3145DB319BB62B /* [CP] Copy Pods Resources */,
+ );
+ buildRules = (
+ );
+ dependencies = (
+ );
+ name = Runner;
+ productName = Runner;
+ productReference = 97C146EE1CF9000F007C117D /* Runner.app */;
+ productType = "com.apple.product-type.application";
+ };
+/* End PBXNativeTarget section */
+
+/* Begin PBXProject section */
+ 97C146E61CF9000F007C117D /* Project object */ = {
+ isa = PBXProject;
+ attributes = {
+ LastUpgradeCheck = 0830;
+ ORGANIZATIONNAME = "The Chromium Authors";
+ TargetAttributes = {
+ 97C146ED1CF9000F007C117D = {
+ CreatedOnToolsVersion = 7.3.1;
+ DevelopmentTeam = A95CP67TEN;
+ };
+ };
+ };
+ buildConfigurationList = 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */;
+ compatibilityVersion = "Xcode 3.2";
+ developmentRegion = English;
+ hasScannedForEncodings = 0;
+ knownRegions = (
+ en,
+ Base,
+ );
+ mainGroup = 97C146E51CF9000F007C117D;
+ productRefGroup = 97C146EF1CF9000F007C117D /* Products */;
+ projectDirPath = "";
+ projectRoot = "";
+ targets = (
+ 97C146ED1CF9000F007C117D /* Runner */,
+ );
+ };
+/* End PBXProject section */
+
+/* Begin PBXResourcesBuildPhase section */
+ 97C146EC1CF9000F007C117D /* Resources */ = {
+ isa = PBXResourcesBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ 9740EEBB1CF902C7004384FC /* app.flx in Resources */,
+ 6E27B1151F0DAFA70028FD65 /* GoogleService-Info.plist in Resources */,
+ 97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */,
+ 9740EEB51CF90195004384FC /* Generated.xcconfig in Resources */,
+ 3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */,
+ 9740EEB41CF90195004384FC /* Debug.xcconfig in Resources */,
+ 97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */,
+ 97C146FC1CF9000F007C117D /* Main.storyboard in Resources */,
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ };
+/* End PBXResourcesBuildPhase section */
+
+/* Begin PBXShellScriptBuildPhase section */
+ 14942D2CEC3145DB319BB62B /* [CP] Copy Pods Resources */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputPaths = (
+ );
+ name = "[CP] Copy Pods Resources";
+ outputPaths = (
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "\"${SRCROOT}/Pods/Target Support Files/Pods-Runner/Pods-Runner-resources.sh\"\n";
+ showEnvVarsInLog = 0;
+ };
+ 2C1A4E9A6849F7CC6B679EA5 /* [CP] Check Pods Manifest.lock */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputPaths = (
+ );
+ name = "[CP] Check Pods Manifest.lock";
+ outputPaths = (
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n";
+ showEnvVarsInLog = 0;
+ };
+ 3B06AD1E1E4923F5004D2608 /* Thin Binary */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputPaths = (
+ );
+ name = "Thin Binary";
+ outputPaths = (
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" thin";
+ };
+ 9740EEB61CF901F6004384FC /* Run Script */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputPaths = (
+ );
+ name = "Run Script";
+ outputPaths = (
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build";
+ };
+ E42D8ABA8B028F5FFACB19B6 /* [CP] Embed Pods Frameworks */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputPaths = (
+ );
+ name = "[CP] Embed Pods Frameworks";
+ outputPaths = (
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "\"${SRCROOT}/Pods/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh\"\n";
+ showEnvVarsInLog = 0;
+ };
+/* End PBXShellScriptBuildPhase section */
+
+/* Begin PBXSourcesBuildPhase section */
+ 97C146EA1CF9000F007C117D /* Sources */ = {
+ isa = PBXSourcesBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ 978B8F6F1D3862AE00F588F7 /* AppDelegate.m in Sources */,
+ 97C146F31CF9000F007C117D /* main.m in Sources */,
+ 1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */,
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ };
+/* End PBXSourcesBuildPhase section */
+
+/* Begin PBXVariantGroup section */
+ 97C146FA1CF9000F007C117D /* Main.storyboard */ = {
+ isa = PBXVariantGroup;
+ children = (
+ 97C146FB1CF9000F007C117D /* Base */,
+ );
+ name = Main.storyboard;
+ sourceTree = "";
+ };
+ 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */ = {
+ isa = PBXVariantGroup;
+ children = (
+ 97C147001CF9000F007C117D /* Base */,
+ );
+ name = LaunchScreen.storyboard;
+ sourceTree = "";
+ };
+/* End PBXVariantGroup section */
+
+/* Begin XCBuildConfiguration section */
+ 97C147031CF9000F007C117D /* Debug */ = {
+ isa = XCBuildConfiguration;
+ baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */;
+ buildSettings = {
+ ALWAYS_SEARCH_USER_PATHS = NO;
+ CLANG_ANALYZER_NONNULL = YES;
+ CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
+ CLANG_CXX_LIBRARY = "libc++";
+ CLANG_ENABLE_MODULES = YES;
+ CLANG_ENABLE_OBJC_ARC = YES;
+ CLANG_WARN_BOOL_CONVERSION = YES;
+ CLANG_WARN_CONSTANT_CONVERSION = YES;
+ CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
+ CLANG_WARN_EMPTY_BODY = YES;
+ CLANG_WARN_ENUM_CONVERSION = YES;
+ CLANG_WARN_INFINITE_RECURSION = YES;
+ CLANG_WARN_INT_CONVERSION = YES;
+ CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
+ CLANG_WARN_SUSPICIOUS_MOVE = YES;
+ CLANG_WARN_UNREACHABLE_CODE = YES;
+ CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
+ "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
+ COPY_PHASE_STRIP = NO;
+ DEBUG_INFORMATION_FORMAT = dwarf;
+ ENABLE_STRICT_OBJC_MSGSEND = YES;
+ ENABLE_TESTABILITY = YES;
+ GCC_C_LANGUAGE_STANDARD = gnu99;
+ GCC_DYNAMIC_NO_PIC = NO;
+ GCC_NO_COMMON_BLOCKS = YES;
+ GCC_OPTIMIZATION_LEVEL = 0;
+ GCC_PREPROCESSOR_DEFINITIONS = (
+ "DEBUG=1",
+ "$(inherited)",
+ );
+ GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
+ GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
+ GCC_WARN_UNDECLARED_SELECTOR = YES;
+ GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
+ GCC_WARN_UNUSED_FUNCTION = YES;
+ GCC_WARN_UNUSED_VARIABLE = YES;
+ IPHONEOS_DEPLOYMENT_TARGET = 8.0;
+ MTL_ENABLE_DEBUG_INFO = YES;
+ ONLY_ACTIVE_ARCH = YES;
+ SDKROOT = iphoneos;
+ TARGETED_DEVICE_FAMILY = "1,2";
+ };
+ name = Debug;
+ };
+ 97C147041CF9000F007C117D /* Release */ = {
+ isa = XCBuildConfiguration;
+ baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
+ buildSettings = {
+ ALWAYS_SEARCH_USER_PATHS = NO;
+ CLANG_ANALYZER_NONNULL = YES;
+ CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
+ CLANG_CXX_LIBRARY = "libc++";
+ CLANG_ENABLE_MODULES = YES;
+ CLANG_ENABLE_OBJC_ARC = YES;
+ CLANG_WARN_BOOL_CONVERSION = YES;
+ CLANG_WARN_CONSTANT_CONVERSION = YES;
+ CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
+ CLANG_WARN_EMPTY_BODY = YES;
+ CLANG_WARN_ENUM_CONVERSION = YES;
+ CLANG_WARN_INFINITE_RECURSION = YES;
+ CLANG_WARN_INT_CONVERSION = YES;
+ CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
+ CLANG_WARN_SUSPICIOUS_MOVE = YES;
+ CLANG_WARN_UNREACHABLE_CODE = YES;
+ CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
+ "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
+ COPY_PHASE_STRIP = NO;
+ DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
+ ENABLE_NS_ASSERTIONS = NO;
+ ENABLE_STRICT_OBJC_MSGSEND = YES;
+ GCC_C_LANGUAGE_STANDARD = gnu99;
+ GCC_NO_COMMON_BLOCKS = YES;
+ GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
+ GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
+ GCC_WARN_UNDECLARED_SELECTOR = YES;
+ GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
+ GCC_WARN_UNUSED_FUNCTION = YES;
+ GCC_WARN_UNUSED_VARIABLE = YES;
+ IPHONEOS_DEPLOYMENT_TARGET = 8.0;
+ MTL_ENABLE_DEBUG_INFO = NO;
+ SDKROOT = iphoneos;
+ TARGETED_DEVICE_FAMILY = "1,2";
+ VALIDATE_PRODUCT = YES;
+ };
+ name = Release;
+ };
+ 97C147061CF9000F007C117D /* Debug */ = {
+ isa = XCBuildConfiguration;
+ baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */;
+ buildSettings = {
+ ARCHS = arm64;
+ ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
+ DEVELOPMENT_TEAM = A95CP67TEN;
+ ENABLE_BITCODE = NO;
+ FRAMEWORK_SEARCH_PATHS = (
+ "$(inherited)",
+ "$(PROJECT_DIR)/Flutter",
+ );
+ INFOPLIST_FILE = Runner/Info.plist;
+ LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
+ LIBRARY_SEARCH_PATHS = (
+ "$(inherited)",
+ "$(PROJECT_DIR)/Flutter",
+ );
+ PRODUCT_BUNDLE_IDENTIFIER = io.flutter.plugins.firebaseAdMobExample;
+ PRODUCT_NAME = "$(TARGET_NAME)";
+ };
+ name = Debug;
+ };
+ 97C147071CF9000F007C117D /* Release */ = {
+ isa = XCBuildConfiguration;
+ baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
+ buildSettings = {
+ ARCHS = arm64;
+ ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
+ DEVELOPMENT_TEAM = A95CP67TEN;
+ ENABLE_BITCODE = NO;
+ FRAMEWORK_SEARCH_PATHS = (
+ "$(inherited)",
+ "$(PROJECT_DIR)/Flutter",
+ );
+ INFOPLIST_FILE = Runner/Info.plist;
+ LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
+ LIBRARY_SEARCH_PATHS = (
+ "$(inherited)",
+ "$(PROJECT_DIR)/Flutter",
+ );
+ PRODUCT_BUNDLE_IDENTIFIER = io.flutter.plugins.firebaseAdMobExample;
+ PRODUCT_NAME = "$(TARGET_NAME)";
+ };
+ name = Release;
+ };
+/* End XCBuildConfiguration section */
+
+/* Begin XCConfigurationList section */
+ 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */ = {
+ isa = XCConfigurationList;
+ buildConfigurations = (
+ 97C147031CF9000F007C117D /* Debug */,
+ 97C147041CF9000F007C117D /* Release */,
+ );
+ defaultConfigurationIsVisible = 0;
+ defaultConfigurationName = Release;
+ };
+ 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */ = {
+ isa = XCConfigurationList;
+ buildConfigurations = (
+ 97C147061CF9000F007C117D /* Debug */,
+ 97C147071CF9000F007C117D /* Release */,
+ );
+ defaultConfigurationIsVisible = 0;
+ defaultConfigurationName = Release;
+ };
+/* End XCConfigurationList section */
+ };
+ rootObject = 97C146E61CF9000F007C117D /* Project object */;
+}
diff --git a/packages/firebase_admob/example/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/packages/firebase_admob/example/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata
new file mode 100644
index 000000000000..1d526a16ed0f
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata
@@ -0,0 +1,7 @@
+
+
+
+
+
diff --git a/packages/firebase_admob/example/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme b/packages/firebase_admob/example/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme
new file mode 100644
index 000000000000..1c9580788197
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme
@@ -0,0 +1,91 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/example/ios/Runner.xcworkspace/contents.xcworkspacedata b/packages/firebase_admob/example/ios/Runner.xcworkspace/contents.xcworkspacedata
new file mode 100644
index 000000000000..21a3cc14c74e
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner.xcworkspace/contents.xcworkspacedata
@@ -0,0 +1,10 @@
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/example/ios/Runner/AppDelegate.h b/packages/firebase_admob/example/ios/Runner/AppDelegate.h
new file mode 100644
index 000000000000..d9e18e990f2e
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/AppDelegate.h
@@ -0,0 +1,10 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import
+#import
+
+@interface AppDelegate : FlutterAppDelegate
+
+@end
diff --git a/packages/firebase_admob/example/ios/Runner/AppDelegate.m b/packages/firebase_admob/example/ios/Runner/AppDelegate.m
new file mode 100644
index 000000000000..f08675707182
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/AppDelegate.m
@@ -0,0 +1,17 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "AppDelegate.h"
+#include "GeneratedPluginRegistrant.h"
+
+@implementation AppDelegate
+
+- (BOOL)application:(UIApplication *)application
+ didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
+ [GeneratedPluginRegistrant registerWithRegistry:self];
+ // Override point for customization after application launch.
+ return [super application:application didFinishLaunchingWithOptions:launchOptions];
+}
+
+@end
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json
new file mode 100644
index 000000000000..d22f10b2ab63
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json
@@ -0,0 +1,116 @@
+{
+ "images" : [
+ {
+ "size" : "20x20",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-20x20@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "20x20",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-20x20@3x.png",
+ "scale" : "3x"
+ },
+ {
+ "size" : "29x29",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-29x29@1x.png",
+ "scale" : "1x"
+ },
+ {
+ "size" : "29x29",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-29x29@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "29x29",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-29x29@3x.png",
+ "scale" : "3x"
+ },
+ {
+ "size" : "40x40",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-40x40@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "40x40",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-40x40@3x.png",
+ "scale" : "3x"
+ },
+ {
+ "size" : "60x60",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-60x60@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "60x60",
+ "idiom" : "iphone",
+ "filename" : "Icon-App-60x60@3x.png",
+ "scale" : "3x"
+ },
+ {
+ "size" : "20x20",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-20x20@1x.png",
+ "scale" : "1x"
+ },
+ {
+ "size" : "20x20",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-20x20@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "29x29",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-29x29@1x.png",
+ "scale" : "1x"
+ },
+ {
+ "size" : "29x29",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-29x29@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "40x40",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-40x40@1x.png",
+ "scale" : "1x"
+ },
+ {
+ "size" : "40x40",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-40x40@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "76x76",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-76x76@1x.png",
+ "scale" : "1x"
+ },
+ {
+ "size" : "76x76",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-76x76@2x.png",
+ "scale" : "2x"
+ },
+ {
+ "size" : "83.5x83.5",
+ "idiom" : "ipad",
+ "filename" : "Icon-App-83.5x83.5@2x.png",
+ "scale" : "2x"
+ }
+ ],
+ "info" : {
+ "version" : 1,
+ "author" : "xcode"
+ }
+}
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png
new file mode 100644
index 000000000000..28c6bf03016f
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png
new file mode 100644
index 000000000000..2ccbfd967d96
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png
new file mode 100644
index 000000000000..f091b6b0bca8
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png
new file mode 100644
index 000000000000..4cde12118dda
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png
new file mode 100644
index 000000000000..d0ef06e7edb8
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png
new file mode 100644
index 000000000000..dcdc2306c285
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png
new file mode 100644
index 000000000000..2ccbfd967d96
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png
new file mode 100644
index 000000000000..c8f9ed8f5cee
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png
new file mode 100644
index 000000000000..a6d6b8609df0
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png
new file mode 100644
index 000000000000..a6d6b8609df0
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png
new file mode 100644
index 000000000000..75b2d164a5a9
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png
new file mode 100644
index 000000000000..c4df70d39da7
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png
new file mode 100644
index 000000000000..6a84f41e14e2
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png
new file mode 100644
index 000000000000..d0e1f5853602
Binary files /dev/null and b/packages/firebase_admob/example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png differ
diff --git a/packages/firebase_admob/example/ios/Runner/Base.lproj/LaunchScreen.storyboard b/packages/firebase_admob/example/ios/Runner/Base.lproj/LaunchScreen.storyboard
new file mode 100644
index 000000000000..ebf48f603974
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/Base.lproj/LaunchScreen.storyboard
@@ -0,0 +1,27 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/example/ios/Runner/Base.lproj/Main.storyboard b/packages/firebase_admob/example/ios/Runner/Base.lproj/Main.storyboard
new file mode 100644
index 000000000000..f3c28516fb38
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/Base.lproj/Main.storyboard
@@ -0,0 +1,26 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/example/ios/Runner/GoogleService-Info.plist b/packages/firebase_admob/example/ios/Runner/GoogleService-Info.plist
new file mode 100644
index 000000000000..907bb24d3bd4
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/GoogleService-Info.plist
@@ -0,0 +1,40 @@
+
+
+
+
+ AD_UNIT_ID_FOR_BANNER_TEST
+ ca-app-pub-3940256099942544/2934735716
+ AD_UNIT_ID_FOR_INTERSTITIAL_TEST
+ ca-app-pub-3940256099942544/4411468910
+ CLIENT_ID
+ 395789065833-eo2pbi7673tisook4bem8jlhikgthqvd.apps.googleusercontent.com
+ REVERSED_CLIENT_ID
+ com.googleusercontent.apps.395789065833-eo2pbi7673tisook4bem8jlhikgthqvd
+ API_KEY
+ AIzaSyAbCza-YTeE5nvWNGGzdlrS4JzZMtfArHY
+ GCM_SENDER_ID
+ 395789065833
+ PLIST_VERSION
+ 1
+ BUNDLE_ID
+ io.flutter.plugins.firebaseadmobile
+ PROJECT_ID
+ admob-test-66963
+ STORAGE_BUCKET
+ admob-test-66963.appspot.com
+ IS_ADS_ENABLED
+
+ IS_ANALYTICS_ENABLED
+
+ IS_APPINVITE_ENABLED
+
+ IS_GCM_ENABLED
+
+ IS_SIGNIN_ENABLED
+
+ GOOGLE_APP_ID
+ 1:395789065833:ios:5c32deb7b961ebcf
+ DATABASE_URL
+ https://admob-test-66963.firebaseio.com
+
+
\ No newline at end of file
diff --git a/packages/firebase_admob/example/ios/Runner/Info.plist b/packages/firebase_admob/example/ios/Runner/Info.plist
new file mode 100644
index 000000000000..6738212b6191
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/Info.plist
@@ -0,0 +1,49 @@
+
+
+
+
+ CFBundleDevelopmentRegion
+ en
+ CFBundleExecutable
+ $(EXECUTABLE_NAME)
+ CFBundleIdentifier
+ $(PRODUCT_BUNDLE_IDENTIFIER)
+ CFBundleInfoDictionaryVersion
+ 6.0
+ CFBundleName
+ firebase_admob_example
+ CFBundlePackageType
+ APPL
+ CFBundleShortVersionString
+ 1.0
+ CFBundleSignature
+ ????
+ CFBundleVersion
+ 1
+ LSRequiresIPhoneOS
+
+ UILaunchStoryboardName
+ LaunchScreen
+ UIMainStoryboardFile
+ Main
+ UIRequiredDeviceCapabilities
+
+ arm64
+
+ UISupportedInterfaceOrientations
+
+ UIInterfaceOrientationPortrait
+ UIInterfaceOrientationLandscapeLeft
+ UIInterfaceOrientationLandscapeRight
+
+ UISupportedInterfaceOrientations~ipad
+
+ UIInterfaceOrientationPortrait
+ UIInterfaceOrientationPortraitUpsideDown
+ UIInterfaceOrientationLandscapeLeft
+ UIInterfaceOrientationLandscapeRight
+
+ UIViewControllerBasedStatusBarAppearance
+
+
+
diff --git a/packages/firebase_admob/example/ios/Runner/main.m b/packages/firebase_admob/example/ios/Runner/main.m
new file mode 100644
index 000000000000..bec320c0bee0
--- /dev/null
+++ b/packages/firebase_admob/example/ios/Runner/main.m
@@ -0,0 +1,13 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import
+#import
+#import "AppDelegate.h"
+
+int main(int argc, char* argv[]) {
+ @autoreleasepool {
+ return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
+ }
+}
diff --git a/packages/firebase_admob/example/lib/main.dart b/packages/firebase_admob/example/lib/main.dart
new file mode 100644
index 000000000000..5d17e66e7e2c
--- /dev/null
+++ b/packages/firebase_admob/example/lib/main.dart
@@ -0,0 +1,122 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+import 'package:flutter/material.dart';
+import 'package:firebase_admob/firebase_admob.dart';
+
+// A placeholder AdMob App Id for testing.
+const String appId = 'ca-app-pub-3940256099942544~3347511713';
+
+// Specify this to quiet Android log messages that say:
+// Use AdRequest.Builder.addTestDevice("...") to get test ads on this device.
+//const String testDevice = null;
+const String testDevice = '33B6FA56617D7688A3A466295DED82BE';
+
+// See https://developers.google.com/admob/ios/test-ads
+const String bannerAdUnitId = 'ca-app-pub-3940256099942544/6300978111';
+const String interstitialAdUnitId = 'ca-app-pub-3940256099942544/1033173712';
+
+class MyApp extends StatefulWidget {
+ @override
+ _MyAppState createState() => new _MyAppState();
+}
+
+class _MyAppState extends State {
+ static final MobileAdTargetingInfo targetingInfo = new MobileAdTargetingInfo(
+ testDevices: testDevice != null ? [testDevice] : null,
+ keywords: ['foo', 'bar'],
+ contentUrl: 'http://foo.com/bar.html',
+ birthday: new DateTime.now(),
+ childDirected: true,
+ gender: MobileAdGender.male,
+ );
+
+ BannerAd _bannerAd;
+ InterstitialAd _interstitialAd;
+
+ BannerAd createBannerAd() {
+ return new BannerAd(
+ unitId: bannerAdUnitId,
+ targetingInfo: targetingInfo,
+ listener: (MobileAdEvent event) {
+ print("BannerAd event $event");
+ },
+ );
+ }
+
+ InterstitialAd createInterstitialAd() {
+ return new InterstitialAd(
+ unitId: interstitialAdUnitId,
+ targetingInfo: targetingInfo,
+ listener: (MobileAdEvent event) {
+ print("InterstitialAd event $event");
+ },
+ );
+ }
+
+ @override
+ void initState() {
+ super.initState();
+ FirebaseAdMob.instance.initialize(appId: appId);
+ _bannerAd = createBannerAd()..load();
+ }
+
+ @override
+ void dispose() {
+ _bannerAd?.dispose();
+ _interstitialAd?.dispose();
+ super.dispose();
+ }
+
+ @override
+ Widget build(BuildContext context) {
+ return new MaterialApp(
+ home: new Scaffold(
+ appBar: new AppBar(
+ title: const Text('AdMob Plugin example app'),
+ ),
+ body: new Center(
+ child: new Column(
+ crossAxisAlignment: CrossAxisAlignment.center,
+ mainAxisSize: MainAxisSize.min,
+ children: [
+ new RaisedButton(
+ child: const Text('SHOW BANNER'),
+ onPressed: () {
+ _bannerAd ??= createBannerAd();
+ _bannerAd
+ ..load()
+ ..show();
+ }),
+ new RaisedButton(
+ child: const Text('REMOVE BANNER'),
+ onPressed: () {
+ _bannerAd?.dispose();
+ _bannerAd = null;
+ }),
+ new RaisedButton(
+ child: const Text('SHOW INTERSTITIAL'),
+ onPressed: () {
+ _interstitialAd?.dispose();
+ _interstitialAd = createInterstitialAd()
+ ..load()
+ ..show();
+ },
+ ),
+ ].map((Widget button) {
+ return new Padding(
+ padding: const EdgeInsets.symmetric(vertical: 16.0),
+ child: button,
+ );
+ }).toList(),
+ ),
+ ),
+ ),
+ );
+ }
+}
+
+void main() {
+ runApp(new MyApp());
+}
diff --git a/packages/firebase_admob/example/pubspec.yaml b/packages/firebase_admob/example/pubspec.yaml
new file mode 100644
index 000000000000..518b50dda0b6
--- /dev/null
+++ b/packages/firebase_admob/example/pubspec.yaml
@@ -0,0 +1,11 @@
+name: firebase_admob_example
+description: Demonstrates how to use the firebase_admob plugin.
+
+dependencies:
+ flutter:
+ sdk: flutter
+ firebase_admob:
+ path: ../
+
+flutter:
+ uses-material-design: true
diff --git a/packages/firebase_admob/firebase_admob.iml b/packages/firebase_admob/firebase_admob.iml
new file mode 100644
index 000000000000..9d5dae19540c
--- /dev/null
+++ b/packages/firebase_admob/firebase_admob.iml
@@ -0,0 +1,15 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/packages/firebase_admob/firebase_admob_android.iml b/packages/firebase_admob/firebase_admob_android.iml
new file mode 100644
index 000000000000..462b903e05b6
--- /dev/null
+++ b/packages/firebase_admob/firebase_admob_android.iml
@@ -0,0 +1,12 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/firebase_admob/ios/.gitignore b/packages/firebase_admob/ios/.gitignore
new file mode 100644
index 000000000000..956c87f3aa28
--- /dev/null
+++ b/packages/firebase_admob/ios/.gitignore
@@ -0,0 +1,31 @@
+.idea/
+.vagrant/
+.sconsign.dblite
+.svn/
+
+.DS_Store
+*.swp
+profile
+
+DerivedData/
+build/
+
+*.pbxuser
+*.mode1v3
+*.mode2v3
+*.perspectivev3
+
+!default.pbxuser
+!default.mode1v3
+!default.mode2v3
+!default.perspectivev3
+
+xcuserdata
+
+*.moved-aside
+
+*.pyc
+*sync/
+Icon?
+.tags*
+
diff --git a/packages/firebase_admob/ios/Assets/.gitkeep b/packages/firebase_admob/ios/Assets/.gitkeep
new file mode 100644
index 000000000000..e69de29bb2d1
diff --git a/packages/firebase_admob/ios/Classes/FirebaseAdmobPlugin.h b/packages/firebase_admob/ios/Classes/FirebaseAdmobPlugin.h
new file mode 100644
index 000000000000..1c32fb4f4955
--- /dev/null
+++ b/packages/firebase_admob/ios/Classes/FirebaseAdmobPlugin.h
@@ -0,0 +1,8 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import
+
+@interface FirebaseAdMobPlugin : NSObject
+@end
diff --git a/packages/firebase_admob/ios/Classes/FirebaseAdmobPlugin.m b/packages/firebase_admob/ios/Classes/FirebaseAdmobPlugin.m
new file mode 100644
index 000000000000..45fd0e945783
--- /dev/null
+++ b/packages/firebase_admob/ios/Classes/FirebaseAdmobPlugin.m
@@ -0,0 +1,142 @@
+// Copyright 2017, the Flutter project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+
+#import "FirebaseAdmobPlugin.h"
+#import "Firebase/Firebase.h"
+#import "MobileAd.h"
+
+@interface NSError (FlutterError)
+@property(readonly, nonatomic) FlutterError *flutterError;
+@end
+
+@implementation NSError (FlutterError)
+- (FlutterError *)flutterError {
+ return [FlutterError errorWithCode:[NSString stringWithFormat:@"%ld", (long)self.code]
+ message:self.domain
+ details:self.localizedDescription];
+}
+@end
+
+@interface FirebaseAdMobPlugin ()
+@property(nonatomic, retain) FlutterMethodChannel *channel;
+@end
+
+@implementation FirebaseAdMobPlugin
+
++ (void)registerWithRegistrar:(NSObject *)registrar {
+ FirebaseAdMobPlugin *instance = [[FirebaseAdMobPlugin alloc] init];
+ instance.channel =
+ [FlutterMethodChannel methodChannelWithName:@"plugins.flutter.io/firebase_admob"
+ binaryMessenger:[registrar messenger]];
+ [registrar addMethodCallDelegate:instance channel:instance.channel];
+}
+
+- (instancetype)init {
+ self = [super init];
+ if (self && ![FIRApp defaultApp]) {
+ NSLog(@"[FIRApp configure]");
+ [FIRApp configure];
+ }
+ return self;
+}
+
+- (void)dealloc {
+ [self.channel setMethodCallHandler:nil];
+ self.channel = nil;
+}
+
+- (void)callInitialize:(FlutterMethodCall *)call result:(FlutterResult)result {
+ NSString *appId = (NSString *)call.arguments[@"appId"];
+ if (appId == nil || [appId length] == 0) {
+ result([FlutterError errorWithCode:@"no_app_id"
+ message:@"a non-empty AdMob appId was not provided"
+ details:nil]);
+ return;
+ }
+ [MobileAd configureWithAppId:appId];
+ result([NSNumber numberWithBool:YES]);
+}
+
+- (void)callLoadAd:(MobileAd *)ad call:(FlutterMethodCall *)call result:(FlutterResult)result {
+ if (ad.status != CREATED) {
+ if (ad.status == FAILED) {
+ NSString *message = [NSString stringWithFormat:@"cannot reload a failed ad=%@", ad];
+ result([FlutterError errorWithCode:@"load_failed_ad" message:message details:nil]);
+ } else {
+ result([NSNumber numberWithBool:YES]); // The ad was already loaded.
+ }
+ }
+
+ NSString *unitId = (NSString *)call.arguments[@"unitId"];
+ if (unitId == nil || [unitId length] == 0) {
+ NSString *message =
+ [NSString stringWithFormat:@"a non-empty unitId was not provided for %@", ad];
+ result([FlutterError errorWithCode:@"no_unit_id" message:message details:nil]);
+ return;
+ }
+
+ NSDictionary *targetingInfo = (NSDictionary *)call.arguments[@"targetingInfo"];
+ [ad loadWithUnitId:unitId targetingInfo:targetingInfo];
+ result([NSNumber numberWithBool:YES]);
+}
+
+- (void)callShowAd:(NSNumber *)mobileAdId
+ call:(FlutterMethodCall *)call
+ result:(FlutterResult)result {
+ MobileAd *ad = [MobileAd getAdForId:mobileAdId];
+ if (ad == nil) {
+ NSString *message =
+ [NSString stringWithFormat:@"show failed, the specified ad was not loaded id=%d",
+ mobileAdId.intValue];
+ result([FlutterError errorWithCode:@"ad_not_loaded" message:message details:nil]);
+ }
+
+ [ad show];
+ result([NSNumber numberWithBool:YES]);
+}
+
+- (void)callDisposeAd:(NSNumber *)mobileAdId
+ call:(FlutterMethodCall *)call
+ result:(FlutterResult)result {
+ MobileAd *ad = [MobileAd getAdForId:mobileAdId];
+ if (ad == nil) {
+ NSString *message =
+ [NSString stringWithFormat:@"dispose failed, no ad exists for id=%d", mobileAdId.intValue];
+ result([FlutterError errorWithCode:@"no_ad_for_id" message:message details:nil]);
+ }
+
+ [ad dispose];
+ result([NSNumber numberWithBool:YES]);
+}
+
+- (void)handleMethodCall:(FlutterMethodCall *)call result:(FlutterResult)result {
+ if ([call.method isEqualToString:@"initialize"]) {
+ [self callInitialize:call result:result];
+ return;
+ }
+
+ NSNumber *mobileAdId = (NSNumber *)call.arguments[@"id"];
+ if (mobileAdId == nil) {
+ NSString *message =
+ @"all FirebaseAdMobPlugin method calls must specify an integer mobile ad id";
+ result([FlutterError errorWithCode:@"no_id" message:message details:nil]);
+ return;
+ }
+
+ if ([call.method isEqualToString:@"loadBannerAd"]) {
+ [self callLoadAd:[BannerAd withId:mobileAdId channel:self.channel] call:call result:result];
+ } else if ([call.method isEqualToString:@"loadInterstitialAd"]) {
+ [self callLoadAd:[InterstitialAd withId:mobileAdId channel:self.channel]
+ call:call
+ result:result];
+ } else if ([call.method isEqualToString:@"showAd"]) {
+ [self callShowAd:mobileAdId call:call result:result];
+ } else if ([call.method isEqualToString:@"disposeAd"]) {
+ [self callDisposeAd:mobileAdId call:call result:result];
+ } else {
+ result(FlutterMethodNotImplemented);
+ }
+}
+
+@end
diff --git a/packages/firebase_admob/ios/Classes/MobileAd.h b/packages/firebase_admob/ios/Classes/MobileAd.h
new file mode 100644
index 000000000000..3b21f4a5de6e
--- /dev/null
+++ b/packages/firebase_admob/ios/Classes/MobileAd.h
@@ -0,0 +1,31 @@
+// Copyright 2017, the Flutter project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+
+#import
+#import "GoogleMobileAds/GoogleMobileAds.h"
+
+typedef enum : NSUInteger {
+ CREATED,
+ LOADING,
+ FAILED,
+ PENDING, // Will be shown when status is changed to LOADED.
+ LOADED,
+} MobileAdStatus;
+
+@interface MobileAd : NSObject
++ (void)configureWithAppId:(NSString *)appId;
++ (MobileAd *)getAdForId:(NSNumber *)mobileAdId;
+- (MobileAdStatus)status;
+- (void)loadWithUnitId:(NSString *)unitId targetingInfo:(NSDictionary *)targetingInfo;
+- (void)show;
+- (void)dispose;
+@end
+
+@interface BannerAd : MobileAd
++ (instancetype)withId:(NSNumber *)mobileAdId channel:(FlutterMethodChannel *)channel;
+@end
+
+@interface InterstitialAd : MobileAd
++ (instancetype)withId:(NSNumber *)mobileAdId channel:(FlutterMethodChannel *)channel;
+@end
diff --git a/packages/firebase_admob/ios/Classes/MobileAd.m b/packages/firebase_admob/ios/Classes/MobileAd.m
new file mode 100644
index 000000000000..4f5e320ff1be
--- /dev/null
+++ b/packages/firebase_admob/ios/Classes/MobileAd.m
@@ -0,0 +1,332 @@
+// Copyright 2017, the Flutter project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+
+#import "MobileAd.h"
+#import "FirebaseAdmobPlugin.h"
+
+static NSMutableDictionary *allAds = nil;
+static NSDictionary *statusToString = nil;
+
+static void logWarning(NSString *format, ...) {
+ va_list args;
+ va_start(args, format);
+ NSString *message = [[NSString alloc] initWithFormat:format arguments:args];
+ va_end(args);
+ NSLog(@"FirebaseAdMobPlugin %@", message);
+}
+
+@implementation MobileAd
+NSNumber *_mobileAdId;
+FlutterMethodChannel *_channel;
+MobileAdStatus _status;
+
++ (void)initialize {
+ if (allAds == nil) {
+ allAds = [[NSMutableDictionary alloc] init];
+ }
+
+ if (statusToString == nil) {
+ statusToString = @{
+ @(CREATED) : @"CREATED",
+ @(LOADING) : @"LOADING",
+ @(FAILED) : @"FAILED",
+ @(PENDING) : @"PENDING",
+ @(LOADED) : @"LOADED"
+ };
+ }
+}
+
++ (void)configureWithAppId:(NSString *)appId {
+ [GADMobileAds configureWithApplicationID:appId];
+}
+
++ (MobileAd *)getAdForId:(NSNumber *)mobileAdId {
+ return allAds[mobileAdId];
+}
+
++ (UIViewController *)rootViewController {
+ return [UIApplication sharedApplication].delegate.window.rootViewController;
+}
+
+- (instancetype)initWithId:(NSNumber *)mobileAdId channel:(FlutterMethodChannel *)channel {
+ self = [super init];
+ if (self) {
+ _mobileAdId = mobileAdId;
+ _channel = channel;
+ _status = CREATED;
+ allAds[mobileAdId] = self;
+ }
+ return self;
+}
+
+- (MobileAdStatus)status {
+ return _status;
+}
+
+- (void)loadWithUnitId:(NSString *)unitId targetingInfo:(NSDictionary *)targetingInfo {
+ // Implemented by the Banner and Interstitial subclasses
+}
+
+- (void)show {
+ // Implemented by the Banner and Interstitial subclasses
+}
+
+- (void)dispose {
+ [allAds removeObjectForKey:_mobileAdId];
+}
+
+- (NSArray *)targetingInfoArrayForKey:(NSString *)key info:(NSDictionary *)info {
+ NSObject *value = info[key];
+ if (value == NULL) {
+ return nil;
+ }
+ if (![value isKindOfClass:[NSArray class]]) {
+ logWarning(@"targeting info %@: expected an array (MobileAd %@)", key, self);
+ return nil;
+ }
+ return (NSArray *)value;
+}
+
+- (NSString *)targetingInfoStringForKey:(NSString *)key info:(NSDictionary *)info {
+ NSObject *value = info[key];
+ if (value == NULL) {
+ return nil;
+ }
+ if (![value isKindOfClass:[NSString class]]) {
+ logWarning(@"targeting info %@: expected a string (MobileAd %@)", key, self);
+ return nil;
+ }
+ NSString *stringValue = (NSString *)value;
+ if ([stringValue length] == 0) {
+ logWarning(@"targeting info %@: expected a non-empty string (MobileAd %@)", key, self);
+ return nil;
+ }
+ return stringValue;
+}
+
+- (NSNumber *)targetingInfoBoolForKey:(NSString *)key info:(NSDictionary *)info {
+ NSObject *value = info[key];
+ if (value == NULL) {
+ return nil;
+ }
+ if (![value isKindOfClass:[NSNumber class]]) {
+ logWarning(@"targeting info %@: expected a boolean, (MobileAd %@)", key, self);
+ return nil;
+ }
+ return (NSNumber *)value;
+}
+
+- (GADRequest *)createLoadRequest:(NSDictionary *)targetingInfo {
+ GADRequest *request = [GADRequest request];
+ if (targetingInfo == nil) {
+ return request;
+ }
+
+ NSArray *testDevices = [self targetingInfoArrayForKey:@"testDevices" info:targetingInfo];
+ if (testDevices != nil) {
+ request.testDevices = testDevices;
+ }
+
+ NSArray *keywords = [self targetingInfoArrayForKey:@"keywords" info:targetingInfo];
+ if (keywords != nil) {
+ request.keywords = keywords;
+ }
+
+ NSString *contentURL = [self targetingInfoStringForKey:@"contentUrl" info:targetingInfo];
+ if (contentURL != nil) {
+ request.contentURL = contentURL;
+ }
+
+ NSObject *birthday = targetingInfo[@"birthday"];
+ if (birthday != NULL) {
+ if (![birthday isKindOfClass:[NSNumber class]]) {
+ logWarning(@"targeting info birthday: expected a long integer (MobileAd %@)", self);
+ } else {
+ // Incoming time value is milliseconds since the epoch, NSDate uses seconds.
+ request.birthday =
+ [NSDate dateWithTimeIntervalSince1970:((NSNumber *)birthday).longValue / 1000.0];
+ }
+ }
+
+ NSObject *gender = targetingInfo[@"gender"];
+ if (gender != NULL) {
+ if (![gender isKindOfClass:[NSNumber class]]) {
+ logWarning(@"targeting info gender: expected an integer (MobileAd %@)", self);
+ } else {
+ int genderValue = ((NSNumber *)gender).intValue;
+ switch (genderValue) {
+ case 0: // MobileAdGender.unknown
+ case 1: // MobileAdGender.male
+ case 2: // MobileAdGender.female
+ request.gender = genderValue;
+ break;
+ default:
+ logWarning(@"targeting info gender: not one of 0, 1, or 2 (MobileAd %@)", self);
+ }
+ }
+ }
+
+ NSNumber *childDirected = [self targetingInfoBoolForKey:@"childDirected" info:targetingInfo];
+ if (childDirected != nil) {
+ [request tagForChildDirectedTreatment:childDirected.boolValue];
+ }
+
+ NSString *requestAgent = [self targetingInfoStringForKey:@"requestAgent" info:targetingInfo];
+ if (requestAgent != nil) {
+ request.requestAgent = requestAgent;
+ }
+
+ return request;
+}
+
+- (NSDictionary *)argumentsMap {
+ return @{@"id" : _mobileAdId};
+}
+
+- (NSString *)description {
+ NSString *statusString = (NSString *)statusToString[[NSNumber numberWithInt:_status]];
+ return [NSString
+ stringWithFormat:@"%@ %@ mobileAdId:%@", super.description, statusString, _mobileAdId];
+}
+@end
+
+@implementation BannerAd
+GADBannerView *_banner;
+
++ (instancetype)withId:(NSNumber *)mobileAdId channel:(FlutterMethodChannel *)channel {
+ MobileAd *ad = [MobileAd getAdForId:mobileAdId];
+ return ad != nil ? (BannerAd *)ad : [[BannerAd alloc] initWithId:mobileAdId channel:channel];
+}
+
+- (void)loadWithUnitId:(NSString *)unitId targetingInfo:(NSDictionary *)targetingInfo {
+ if (_status != CREATED) return;
+ _status = LOADING;
+ _banner = [[GADBannerView alloc] initWithAdSize:kGADAdSizeBanner];
+ _banner.delegate = self;
+ _banner.adUnitID = unitId;
+ _banner.rootViewController = [MobileAd rootViewController];
+ [_banner loadRequest:[self createLoadRequest:targetingInfo]];
+}
+
+- (void)show {
+ if (_status == LOADING) {
+ _status = PENDING;
+ return;
+ }
+ if (_status != LOADED) return;
+
+ UIView *screen = [MobileAd rootViewController].view;
+ CGFloat x = screen.frame.size.width / 2 - _banner.frame.size.width / 2;
+ CGFloat y = screen.frame.size.height - _banner.frame.size.height;
+ _banner.frame = (CGRect){{x, y}, _banner.frame.size};
+ [screen addSubview:_banner];
+}
+
+- (void)adViewDidReceiveAd:(GADBannerView *)adView {
+ bool statusWasPending = _status == PENDING;
+ _status = LOADED;
+ [_channel invokeMethod:@"onAdLoaded" arguments:[self argumentsMap]];
+ if (statusWasPending) [self show];
+}
+
+- (void)adView:(GADBannerView *)adView didFailToReceiveAdWithError:(GADRequestError *)error {
+ logWarning(@"adView:didFailToReceiveAdWithError: %@ (MobileAd %@)", [error localizedDescription],
+ self);
+ [_channel invokeMethod:@"onAdFailedToLoad" arguments:[self argumentsMap]];
+}
+
+- (void)adViewWillPresentScreen:(GADBannerView *)adView {
+ [_channel invokeMethod:@"onAdClicked" arguments:[self argumentsMap]];
+}
+
+- (void)adViewWillDismissScreen:(GADBannerView *)adView {
+ [_channel invokeMethod:@"onAdImpression" arguments:[self argumentsMap]];
+}
+
+- (void)adViewDidDismissScreen:(GADBannerView *)adView {
+ [_channel invokeMethod:@"onAdClosed" arguments:[self argumentsMap]];
+}
+
+- (void)adViewWillLeaveApplication:(GADBannerView *)adView {
+ [_channel invokeMethod:@"onAdLeftApplication" arguments:[self argumentsMap]];
+}
+
+- (void)dispose {
+ if (_banner.superview) [_banner removeFromSuperview];
+ _banner = nil;
+ [super dispose];
+}
+
+- (NSString *)description {
+ return [NSString stringWithFormat:@"%@ for: %@", super.description, _banner];
+}
+@end
+
+@implementation InterstitialAd
+GADInterstitial *_interstitial;
+
++ (instancetype)withId:(NSNumber *)mobileAdId channel:(FlutterMethodChannel *)channel {
+ MobileAd *ad = [MobileAd getAdForId:mobileAdId];
+ return ad != nil ? (InterstitialAd *)ad
+ : [[InterstitialAd alloc] initWithId:mobileAdId channel:channel];
+}
+
+- (void)loadWithUnitId:(NSString *)unitId targetingInfo:(NSDictionary *)targetingInfo {
+ if (_status != CREATED) return;
+ _status = LOADING;
+
+ _interstitial = [[GADInterstitial alloc] initWithAdUnitID:unitId];
+ _interstitial.delegate = self;
+ [_interstitial loadRequest:[self createLoadRequest:targetingInfo]];
+}
+
+- (void)show {
+ if (_status == LOADING) {
+ _status = PENDING;
+ return;
+ }
+ if (_status != LOADED) return;
+
+ [_interstitial presentFromRootViewController:[MobileAd rootViewController]];
+}
+
+- (void)interstitialDidReceiveAd:(GADInterstitial *)ad {
+ bool statusWasPending = _status == PENDING;
+ _status = LOADED;
+ [_channel invokeMethod:@"onAdLoaded" arguments:[self argumentsMap]];
+ if (statusWasPending) [self show];
+}
+
+- (void)interstitial:(GADInterstitial *)ad didFailToReceiveAdWithError:(GADRequestError *)error {
+ logWarning(@"interstitial:didFailToReceiveAdWithError: %@ (MobileAd %@)",
+ [error localizedDescription], self);
+ [_channel invokeMethod:@"onAdFailedToLoad" arguments:[self argumentsMap]];
+}
+
+- (void)interstitialWillPresentScreen:(GADInterstitial *)ad {
+ [_channel invokeMethod:@"onAdClicked" arguments:[self argumentsMap]];
+}
+
+- (void)interstitialWillDismissScreen:(GADInterstitial *)ad {
+ [_channel invokeMethod:@"onAdImpression" arguments:[self argumentsMap]];
+}
+
+- (void)interstitialDidDismissScreen:(GADInterstitial *)ad {
+ [_channel invokeMethod:@"onAdClosed" arguments:[self argumentsMap]];
+}
+
+- (void)interstitialWillLeaveApplication:(GADInterstitial *)ad {
+ [_channel invokeMethod:@"onAdLeftApplication" arguments:[self argumentsMap]];
+}
+
+- (void)dispose {
+ // It is not possible to hide/remove/destroy an AdMob interstitial Ad.
+ _interstitial = nil;
+ [super dispose];
+}
+
+- (NSString *)description {
+ return [NSString stringWithFormat:@"%@ for: %@", super.description, _interstitial];
+}
+@end
diff --git a/packages/firebase_admob/ios/firebase_admob.podspec b/packages/firebase_admob/ios/firebase_admob.podspec
new file mode 100644
index 000000000000..c549b5e06763
--- /dev/null
+++ b/packages/firebase_admob/ios/firebase_admob.podspec
@@ -0,0 +1,22 @@
+#
+# To learn more about a Podspec see http://guides.cocoapods.org/syntax/podspec.html
+#
+Pod::Spec.new do |s|
+ s.name = 'firebase_admob'
+ s.version = '0.0.1'
+ s.summary = 'Firebase Admob plugin for Flutter.'
+ s.description = <<-DESC
+Firebase Admob plugin for Flutter.
+ DESC
+ s.homepage = 'https://github.com/flutter/plugins/tree/master/packages/firebase_admob'
+ s.license = { :file => '../LICENSE' }
+ s.author = { 'Flutter Team' => 'flutter-dev@googlegroups.com' }
+ s.source = { :path => '.' }
+ s.source_files = 'Classes/**/*'
+ s.public_header_files = 'Classes/**/*.h'
+ s.dependency 'Flutter'
+ s.dependency 'Firebase/Core'
+ s.dependency 'Firebase/AdMob'
+
+ s.ios.deployment_target = '8.0'
+end
diff --git a/packages/firebase_admob/lib/firebase_admob.dart b/packages/firebase_admob/lib/firebase_admob.dart
new file mode 100644
index 000000000000..9e66a8718841
--- /dev/null
+++ b/packages/firebase_admob/lib/firebase_admob.dart
@@ -0,0 +1,258 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+import 'dart:async';
+
+import 'package:flutter/services.dart';
+import 'package:meta/meta.dart';
+
+/// [MobileAd] status changes reported to [MobileAdListener]s.
+///
+/// Applications can wait until an ad is [MobileAdEvent.loaded] before showing
+/// it, to ensure that the ad is displayed promptly.
+enum MobileAdEvent {
+ loaded,
+ failedToLoad,
+ clicked,
+ impression,
+ opened,
+ leftApplication,
+ closed,
+}
+
+/// The user's gender for the sake of ad targeting using [MobileAdTargetingInfo].
+// Warning: the index values of the enums must match the values of the corresponding
+// AdMob constants. For example MobileAdGender.female.index == kGADGenderFemale.
+enum MobileAdGender {
+ unknown,
+ male,
+ female,
+}
+
+/// Signature for a [MobileAd] status change callback.
+typedef void MobileAdListener(MobileAdEvent event);
+
+/// Targeting info per the native AdMob API.
+///
+/// This class's properties mirror the native AdRequest API. See for example:
+/// [AdRequest.Builder for Android](https://firebase.google.com/docs/reference/android/com/google/android/gms/ads/AdRequest.Builder).
+class MobileAdTargetingInfo {
+ const MobileAdTargetingInfo({
+ this.keywords,
+ this.contentUrl,
+ this.birthday,
+ this.gender,
+ this.designedForFamilies,
+ this.childDirected,
+ this.testDevices,
+ this.requestAgent,
+ });
+
+ final List keywords;
+ final String contentUrl;
+ final DateTime birthday;
+ final MobileAdGender gender;
+ final bool designedForFamilies;
+ final bool childDirected;
+ final List testDevices;
+ final String requestAgent;
+
+ Map toJson() {
+ final Map json = {};
+
+ if (keywords != null && keywords.isNotEmpty) {
+ assert(keywords.every((String s) => s != null && s.isNotEmpty));
+ json['keywords'] = keywords;
+ }
+ if (contentUrl != null && contentUrl.isNotEmpty)
+ json['contentUrl'] = contentUrl;
+ if (birthday != null) json['birthday'] = birthday.millisecondsSinceEpoch;
+ if (gender != null) json['gender'] = gender.index;
+ if (designedForFamilies != null)
+ json['designedForFamilies'] = designedForFamilies;
+ if (childDirected != null) json['childDirected'] = childDirected;
+ if (testDevices != null && testDevices.isNotEmpty) {
+ assert(testDevices.every((String s) => s != null && s.isNotEmpty));
+ json['testDevices'] = testDevices;
+ }
+ if (requestAgent != null && requestAgent.isNotEmpty)
+ json['requestAgent'] = requestAgent;
+
+ return json;
+ }
+}
+
+/// A mobile [BannerAd] or [InterstitialAd] for the [FirebaseAdMobPlugin].
+///
+/// A [MobileAd] must be loaded with [load] before it is shown with [show].
+///
+/// A valid [unitId] is required.
+abstract class MobileAd {
+ static final Map _allAds = {};
+
+ /// Default constructor, used by subclasses.
+ MobileAd({@required this.unitId, this.targetingInfo, this.listener}) {
+ assert(unitId != null && unitId.isNotEmpty);
+ assert(_allAds[id] == null);
+ _allAds[id] = this;
+ }
+
+ /// Optional targeting info per the native AdMob API.
+ final MobileAdTargetingInfo targetingInfo;
+
+ /// Identifies the source of ads for your application.
+ ///
+ /// For testing use a [sample ad unit](https://developers.google.com/admob/ios/test-ads#sample_ad_units).
+ final String unitId;
+
+ /// Called when the status of the ad changes.
+ final MobileAdListener listener;
+
+ /// An internal id that identifies this mobile ad to the native AdMob plugin.
+ ///
+ /// Plugin log messages will identify this property as the ad's `mobileAdId`.
+ int get id => hashCode;
+
+ MethodChannel get _channel => FirebaseAdMob.instance._channel;
+
+ /// Start loading this ad.
+ Future load();
+
+ /// Show this ad.
+ ///
+ /// The ad must have been loaded with [load] first. If loading hasn't finished
+ /// the ad will not actually appear until the ad has finished loading.
+ ///
+ /// The [listener] will be notified when the ad has finished loading or fails
+ /// to do so. An ad that fails to load will not be shown.
+ Future show() {
+ return _channel.invokeMethod("showAd", {'id': id});
+ }
+
+ /// Free the plugin resources associated with this ad.
+ ///
+ /// Disposing a banner ad that's been shown removes it from the screen. Interstitial
+ /// ads can't be programatically removed from view.
+ Future dispose() {
+ assert(_allAds[id] != null);
+ _allAds[id] = null;
+ return _channel.invokeMethod("disposeAd", {'id': id});
+ }
+
+ Future _doLoad(String loadMethod) {
+ return _channel.invokeMethod(loadMethod, {
+ 'id': id,
+ 'unitId': unitId,
+ 'targetingInfo': targetingInfo?.toJson(),
+ });
+ }
+}
+
+/// A banner ad for the [FirebaseAdMobPlugin].
+class BannerAd extends MobileAd {
+ /// Create a BannerAd.
+ ///
+ /// A valid [unitId] is required.
+ BannerAd({
+ @required String unitId,
+ MobileAdTargetingInfo targetingInfo,
+ MobileAdListener listener,
+ })
+ : super(unitId: unitId, targetingInfo: targetingInfo, listener: listener);
+
+ @override
+ Future load() => _doLoad("loadBannerAd");
+}
+
+/// A full-screen interstitial ad for the [FirebaseAdMobPlugin].
+class InterstitialAd extends MobileAd {
+ /// Create an Interstitial.
+ ///
+ /// A valid [unitId] is required.
+ InterstitialAd({
+ String unitId,
+ MobileAdTargetingInfo targetingInfo,
+ MobileAdListener listener,
+ })
+ : super(unitId: unitId, targetingInfo: targetingInfo, listener: listener);
+
+ @override
+ Future load() => _doLoad("loadInterstitialAd");
+}
+
+/// Support for Google AdMob mobile ads.
+///
+/// Before loading or showing an ad the plugin must be initialized with
+/// an AdMob app id:
+/// ```
+/// FirebaseAdMob.instance.initialize(appId: myAppId);
+/// ```
+///
+/// Apps can create, load, and show mobile ads. For example:
+/// ```
+/// BannerAd myBanner = new BannerAd(unitId: myBannerAdUnitId)
+/// ..load()
+/// ..show();
+/// ```
+///
+/// See also:
+///
+/// * The example associated with this plugin.
+/// * [BannerAd], a small rectangular ad displayed at the bottom of the screen.
+/// * [InterstitialAd], a full screen ad that must be dismissed by the user.
+class FirebaseAdMob {
+ @visibleForTesting
+ FirebaseAdMob.private(MethodChannel channel) : _channel = channel {
+ _channel.setMethodCallHandler(_handleMethod);
+ }
+
+ static final FirebaseAdMob _instance = new FirebaseAdMob.private(
+ const MethodChannel('plugins.flutter.io/firebase_admob'),
+ );
+
+ /// The single shared instance of this plugin.
+ static FirebaseAdMob get instance => _instance;
+
+ final MethodChannel _channel;
+
+ static const Map _methodToEvent =
+ const {
+ 'onAdLoaded': MobileAdEvent.loaded,
+ 'onAdFailedToLoad': MobileAdEvent.failedToLoad,
+ 'onAdClicked': MobileAdEvent.clicked,
+ 'onAdImpression': MobileAdEvent.impression,
+ 'onAdOpened': MobileAdEvent.opened,
+ 'onAdLeftApplication': MobileAdEvent.leftApplication,
+ 'onAdClosed': MobileAdEvent.closed,
+ };
+
+ /// Initialize this plugin for the AdMob app specified by `appId`.
+ ///
+ /// For testing one can use `ca-app-pub-3940256099942544~3347511713` for the `appId`.
+ Future initialize(
+ {@required String appId,
+ String trackingId,
+ bool analyticsEnabled: false}) {
+ assert(appId != null && appId.isNotEmpty);
+ assert(analyticsEnabled != null);
+ return _channel.invokeMethod("initialize", {
+ 'appId': appId,
+ 'trackingId': trackingId,
+ 'analyticsEnabled': analyticsEnabled,
+ });
+ }
+
+ Future _handleMethod(MethodCall call) {
+ assert(call.arguments is Map);
+ final Map argumentsMap = call.arguments;
+ final int id = argumentsMap['id'];
+ if (id != null && MobileAd._allAds[id] != null) {
+ final MobileAd ad = MobileAd._allAds[id];
+ final MobileAdEvent event = _methodToEvent[call.method];
+ if (event != null && ad.listener != null) ad.listener(event);
+ }
+
+ return new Future(null);
+ }
+}
diff --git a/packages/firebase_admob/pubspec.yaml b/packages/firebase_admob/pubspec.yaml
new file mode 100644
index 000000000000..ed485b554118
--- /dev/null
+++ b/packages/firebase_admob/pubspec.yaml
@@ -0,0 +1,23 @@
+name: firebase_admob
+description: Firebase AdMob plugin for Flutter applications.
+version: 0.0.1
+author: Flutter Team
+homepage: https://github.com/flutter/plugins/tree/master/packages/firebase_admob
+
+flutter:
+ plugin:
+ androidPackage: io.flutter.plugins.firebaseadmob
+ pluginClass: FirebaseAdMobPlugin
+
+dependencies:
+ meta: ^1.0.4
+ platform: ^2.0.0
+ flutter:
+ sdk: flutter
+
+dev_dependencies:
+ mockito: ^2.0.2
+ test: 0.12.21
+
+environment:
+ sdk: ">=1.8.0 <2.0.0"
diff --git a/packages/firebase_admob/test/firebase_admob_test.dart b/packages/firebase_admob/test/firebase_admob_test.dart
new file mode 100644
index 000000000000..a81619d79481
--- /dev/null
+++ b/packages/firebase_admob/test/firebase_admob_test.dart
@@ -0,0 +1,112 @@
+// Copyright 2017 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+import 'dart:async';
+import 'package:flutter/services.dart';
+import 'package:firebase_admob/firebase_admob.dart';
+import 'package:test/test.dart';
+
+void main() {
+ group('FirebaseAdMob', () {
+ const MethodChannel channel =
+ const MethodChannel('plugins.flutter.io/firebase_admob');
+
+ const String appId = 'ca-app-pub-3940256099942544~3347511713';
+ const String bannerAdUnitId = 'ca-app-pub-3940256099942544/6300978111';
+ const String interstitialAdUnitId =
+ 'ca-app-pub-3940256099942544/1033173712';
+
+ final List log = [];
+ final FirebaseAdMob admob = new FirebaseAdMob.private(channel);
+
+ setUp(() async {
+ channel.setMockMethodCallHandler((MethodCall methodCall) async {
+ log.add(methodCall);
+ switch (methodCall.method) {
+ case 'initialize':
+ case 'loadBannerAd':
+ case 'loadInterstitialAd':
+ case 'showAd':
+ case 'disposeAd':
+ return new Future.value(true);
+ default:
+ assert(false);
+ }
+ });
+ });
+
+ test('initialize', () async {
+ log.clear();
+
+ expect(await admob.initialize(appId: appId), true);
+ expect(
+ log,
+ equals([
+ new MethodCall('initialize', {
+ 'appId': appId,
+ 'trackingId': null,
+ 'analyticsEnabled': false,
+ }),
+ ]));
+ });
+
+ test('banner', () async {
+ log.clear();
+
+ final BannerAd banner = new BannerAd(
+ unitId: bannerAdUnitId,
+ );
+ final int id = banner.id;
+
+ expect(await banner.load(), true);
+ expect(await banner.show(), true);
+ expect(await banner.dispose(), true);
+
+ expect(
+ log,
+ equals([
+ new MethodCall('loadBannerAd', {
+ 'id': id,
+ 'unitId': bannerAdUnitId,
+ 'targetingInfo': null,
+ }),
+ new MethodCall('showAd', {
+ 'id': id,
+ }),
+ new MethodCall('disposeAd', {
+ 'id': id,
+ }),
+ ]));
+ });
+
+ test('interstitial', () async {
+ log.clear();
+
+ final InterstitialAd interstitial = new InterstitialAd(
+ unitId: interstitialAdUnitId,
+ );
+ final int id = interstitial.id;
+
+ expect(await interstitial.load(), true);
+ expect(await interstitial.show(), true);
+ expect(await interstitial.dispose(), true);
+
+ expect(
+ log,
+ equals([
+ new MethodCall('loadInterstitialAd', {
+ 'id': id,
+ 'unitId': interstitialAdUnitId,
+ 'targetingInfo': null,
+ }),
+ new MethodCall('showAd', {
+ 'id': id,
+ }),
+ new MethodCall('disposeAd', {
+ 'id': id,
+ }),
+ ]));
+ });
+ });
+}