# ADSuyi广告聚合SdkDemo_flutter_sdk **Repository Path**: admobile/ADSuyiSdkDemo-flutter ## Basic Information - **Project Name**: ADSuyi广告聚合SdkDemo_flutter_sdk - **Description**: ADmobile ADSuyi-Flutter 插件 ADmobile广告聚合平台,​让移动开发者可以根据自身业务选择支持接入的三方广告平台,并能快速接入、统一管理广告流量,同时借助ADmobile广告数据分析和聚合优化工具提升收益。SDK支持一站聚合管理优量汇穿山甲快手联盟等国内十几家主流移动广告平台,提供开屏广告、原生广告、插屏广告、横幅广告、激励视频广告、视频广告、全屏视频、插屏视频等广告样式,满足各类移动广告变现需求。支持Android、iOS、Flutter开发。 - **Primary Language**: Dart - **License**: GPL-3.0 - **Default Branch**: master - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 27 - **Forks**: 12 - **Created**: 2021-05-10 - **Last Updated**: 2025-06-16 ## Categories & Tags **Categories**: mobile-develop **Tags**: None ## README # Suyi聚合广告SDK Flutter 插件 [TOC] ## 1.1 概述 尊敬的开发者朋友,欢迎您使用Suyi聚合广告SDK。通过本文档,您可以在几分钟之内轻松完成广告的集成过程。 操作系统: iOS 9.0 及以上版本,Android 4.4 及以上版本, 运行设备:iPhone (iPad上可能部分广告正常展示,但是存在填充很低或者平台不支持等问题,建议不要在iPad上展示广告),Android ## 2.1 SDK导入 首先需要导入主SDK ```dart dependencies: ad_suyi_flutter_sdk: {library version} ``` 然后需要导入各平台SDK, ### 2.1.1 iOS在项目的podfile中增加如下内容,可以根据实际需要选择性导入平台 ```ruby pod 'ADSuyiSDK','~> 4.0.0.04221' # 主SDK 必选 pod 'ADSuyiSDK/ADSuyiSDKPlatforms/tianmu' # 天目 #必选 pod 'ADSuyiSDK/ADSuyiSDKPlatforms/gdt' # 优量汇(广点通) pod 'ADSuyiSDK/ADSuyiSDKPlatforms/baidu' # 百度 pod 'ADSuyiSDK/ADSuyiSDKPlatforms/ks' # 快手 pod 'ADSuyiSDK/ADSuyiSDKPlatforms/jad' # 京媒,白名单需添加京东openapp.jdmobile pod 'ADSuyiSDK/ADSuyiSDKPlatforms/iqy' # 爱奇艺 # ab二选一 # a.不需要gromore pod 'ADSuyiSDK/ADSuyiSDKPlatforms/bu' # 穿山甲(头条) # b.需要gromore # pod 'ADSuyiSDK/ADSuyiSDKPlatforms/bu-without' # 穿山甲(头条) # pod 'ADSuyiSDK/ADSuyiSDKPlatforms/gromore' # gromore ``` ### 2.1.2 Android在项目中增加如下内容,可以根据实际需要选择性导入平台 #### 2.1.2.1. 在android根目录build.gradle中添加suyi仓库 ```java allprojects { repositories { ... google() jcenter() mavenCentral() // 添加以下仓库地址 // ADSuyi远程仓库 maven { url "https://maven.admobile.top/repository/maven-releases/" } // 如果添加了云码广告,需要添加云码的远程仓库依赖 maven { url 'http://maven.aliyun.com/nexus/content/repositories/releases/' } // 如果添加了华为联盟广告,需要添加华为联盟的远程仓库依赖 maven { url 'https://developer.huawei.com/repo/' } } } ``` #### 2.1.2.2. OAID支持 导入安全联盟的OAID支持库 oaid_sdk_1.0.25.aar,可在Demo的libs目录下找到,强烈建议使用和Demo中一样版本的OAID库(包括项目中已存在的依赖的oaid版本); 将Demo中assets文件夹下的supplierconfig.json文件复制到自己的assets目录下并按照supplierconfig.json文件中的说明进行OAID的 AppId 配置,supplierconfig.json文件名不可修改; #### 2.1.2.3. 在android/app目录build.gradle中添加相关依赖 ```java // support支持库,如果是AndroidX请使用对应的支持库 implementation 'com.android.support:appcompat-v7:28.0.0' implementation 'com.android.support:support-v4:28.0.0' implementation 'com.android.support:design:28.0.0' // ADSuyiSdk核心库必须导入 implementation 'cn.admobiletop.adsuyi.ad:core:4.0.0.05062' // 天目适配器,必选 implementation 'cn.admobiletop.adsuyi.ad.adapter:tianmu:3.0.0.05151' // 优量汇(广点通)适配器,可选 implementation 'cn.admobiletop.adsuyi.ad.adapter:gdt:4.640.1510.05133' // 穿山甲(头条)适配器,可选 implementation 'cn.admobiletop.adsuyi.ad.adapter:toutiao:6.8.1.7.05132' // 百度适配器,可选。仅支持AndroidX版本 implementation 'cn.admobiletop.adsuyi.ad.adapter:baidu-androidx:9.391.05141' // 快手适配器,可选 implementation 'cn.admobiletop.adsuyi.ad.adapter:ksadbase:3.3.76.5.05141' // 米盟适配器,可选。仅支持AndroidX版本 implementation 'cn.admobiletop.adsuyi.ad.adapter:mimo-androidx:5.3.3.12103' implementation 'com.google.code.gson:gson:2.8.5' implementation 'com.github.bumptech.glide:glide:4.9.0' annotationProcessor 'com.github.bumptech.glide:compiler:4.9.0' // 华为适配器,可选 implementation 'cn.admobiletop.adsuyi.ad.adapter:hwpps:13.4.76.300.01031' // 京媒适配器,可选。仅支持AndroidX版本 implementation 'cn.admobiletop.adsuyi.ad.adapter:jadyun-androidx:2.6.28.05141' // 章鱼适配器,可选的 implementation 'cn.admobiletop.adsuyi.ad.adapter:octopus:1.6.3.6.05141' // 爱奇艺适配器,可选 implementation 'cn.admobiletop.adsuyi.ad.adapter:iqy:1.15.100.53.05141' // gromoreAdapterSdk,可选的。如使用gromore的其他渠道,请联系开发者。 // gromore已与穿山甲合并,此处无需重复导入穿山甲。 implementation 'cn.admobiletop.adsuyi.ad.adapter:gromore-without:6.6.0.7.01084' implementation "com.pangle.cn:mediation-gdt-adapter:4.575.1445.1" //广点通 adapter // 有gromore其他渠道需求,请联系开发者。 ``` #### 2.1.2.4. 广告位创建注意事项 由于使用flutter方式对接,信息流广告可能存在无法展示广告问题,需要在创建优量汇、百度、快手渠道时,创建接近10:9比例的广告样式,可以达到最好的展示效果,否则会出现留白情况。 ## 3.1 工程环境配置 ### 3.1.1 IOS 工程环境配置 #### 3.1.1.1. info.plist 添加支持 Http访问字段 ``` NSAppTransportSecurity NSAllowsArbitraryLoads ``` #### 3.1.1.2. Info.plist 添加定位权限字段(使用ADMobGenLocation可不设置) ``` NSLocationWhenInUseUsageDescription NSLocationAlwaysAndWhenInUseUsageDeion ``` #### 3.1.1.3. Info.plist 添加获取本地网络权限字段 ``` Privacy - Local Network Usage Description 广告投放及广告监测归因、反作弊 Bonjour services _apple-mobdev2._tcp.local ``` #### 3.1.1.4. Info.plist推荐设置白名单,可提高广告收益 ``` LSApplicationQueriesSchemes com.suning.SuningEBuy openapp.jdmobile openjd jdmobile vmall vipshop suning yohobuy kaola yanxuan wbmain dianping imeituan beibeiapp taobao tmall wireless1688 tbopen taobaolite taobaoliveshare koubei eleme alipays kfcapplinkurl pddopen pinduoduo mogujie lianjiabeike lianjia openanjuke zhuanzhuan farfetchCN weibo xhsdiscover uclink momochat blued zhihu baiduboxapp yidui sinanews snssdk141 newsapp igetApp kuaikan youku snssdk1128 gifshow snssdk1112 miguvideo iqiyi bilibili tenvideo baiduhaokan yykiwi qqmusic orpheus kugouURL qmkege changba iting ctrip QunarAlipay diditaxi didicommon taobaotravel OneTravel kfhxztravel gengmei app.soyoung ``` ### 3.1.2 Android 工程环境配置 #### 3.1.2.1 权限配置 ```java ``` #### 3.1.2.2 FileProvider配置 适配Anroid7.0以及以上,请在AndroidManifest中添加如下代码: 如果支持库是support ```java ``` 如果支持库为androidx ```java ``` 在res/xml目录下(如果xml目录不存在需要手动创建),新建xml文件adsuyi_file_paths,在该文件中加入如下配置,如果存在相同android:authorities的provider,请将paths标签中的路劲配置到自己的xml文件中: ```java ``` #### 3.1.2.3 网络配置 需要在 AndroidManifest.xml 添加依赖声明uses-library android:name="org.apache.http.legacy" android:required="false", 且 application标签中添加 android:usesCleartextTraffic="true",适配网络http请求,否则 SDK可能无法正常工作,接入代码示例如下: ```java ... ... ``` #### 3.1.2.4 混淆配置 可以参考demo中proguard-rules.pro相关配置 #### 3.1.2.5 注册java与flutter交互代码 [参考资料](https://developers.google.cn/ad-manager/mobile-ads-sdk/flutter/native/get-started?hl=zh-cn) ```java public class MainActivity extends FlutterActivity { ... @Override public void configureFlutterEngine(FlutterEngine flutterEngine) { super.configureFlutterEngine(flutterEngine); ... Log.d("configureFlutterEngine", "init"); flutterEngine.getPlugins().add(new ADSuyiMobileAdsPlugin()); ADSuyiMobileAdsPlugin.registerNativeAdFactory(flutterEngine, "adFactoryExample"); ... } @Override public void cleanUpFlutterEngine(FlutterEngine flutterEngine) { } ... } ``` #### 3.1.2.6 导入插件后的异常处理 Q:Plugin project :ad_suyi_flutter_sdk not found. Please update settings.gradle A:可参考demo中的settings.gradle,将配置拷贝到项目中 ## 3.2 iOS14适配 由于iOS14中对于权限和隐私内容有一定程度的修改,而且和广告业务关系较大,请按照如下步骤适配,如果未适配。不会导致运行异常或者崩溃等情况,但是会一定程度上影响广告收入。敬请知悉。 1. 应用编译环境升级至 Xcode 12.0 及以上版本; 2. 升级ADSuyiSDK 3.1.5及以上版本; 3. 设置SKAdNetwork和IDFA权限; ### 3.2.1 获取App Tracking Transparency授权(弹窗授权获取IDFA) 从 iOS 14 开始,在应用程序调用 App Tracking Transparency 向用户提跟踪授权请求之前,IDFA 将不可用。 1. 更新 Info.plist,添加 NSUserTrackingUsageDescription 字段和自定义文案描述。 弹窗小字文案建议: - `获取标记权限向您提供更优质、安全的个性化服务及内容,未经同意我们不会用于其他目的;开启后,您也可以前往系统“设置-隐私 ”中随时关闭。` - `获取IDFA标记权限向您提供更优质、安全的个性化服务及内容;开启后,您也可以前往系统“设置-隐私 ”中随时关闭。` ``` NSUserTrackingUsageDescription 获取标记权限向您提供更优质、安全的个性化服务及内容,未经同意我们不会用于其他目的;开启后,您也可以前往系统“设置-隐私 ”中随时关闭 ``` 1. 向用户申请权限。 ``` #import #import ... - (void)requestIDFA { [ATTrackingManager requestTrackingAuthorizationWithCompletionHandler:^(ATTrackingManagerAuthorizationStatus status) { // 无需对授权状态进行处理 }]; } // 建议启动App用户同意协议后就获取权限或者请求广告前获取 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // 针对iOS15中不弹窗被拒解决方案,方案1:经测试可能无效 //dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ // 用户同意协议后获取 //[self requestIDFA]; //}); } // 方案2:根据官方文档调整权限申请时机 // 根据官方开发文档选择在此方法中进行权限申请 - (void)applicationDidBecomeActive:(UIApplication *)application { // 用户同意协议后获取 [self requestIDFA]; } // 建议方案1与2一起使用,可正常通过审核。 ``` ### 3.2.2 SKAdNetwork SKAdNetwork 是接收iOS端营销推广活动归因数据的一种方法。 1. 将下列SKAdNetwork ID(供参考,需以穿山甲官网为准)添加到 info.plist 中,以保证 SKAdNetwork 的正确运行。根据对接平台添加相应SKAdNetworkID,若无对接平台SKNetworkID则无需添加。 ``` SKAdNetworkItems // 穿山甲广告(ADSuyiBU) SKAdNetworkIdentifier 238da6jt44.skadnetwork SKAdNetworkIdentifier 22mmun2rn5.skadnetwork SKAdNetworkIdentifier x2jnk7ly8j.skadnetwork ``` ## 4.1 主SDK初始化 ```dart AdSuyiFlutterSdk.initSdk(appid: "appid"); ``` ## 4.2 开屏广告 ### 4.2.1 开屏广告(跳转至原生页面加载并展示) ```dart class SplashPage extends StatefulWidget { @override State createState() => _SplashState(); } class _SplashState extends State { ADSuyiSplashAd _splashAd; @override Widget build(BuildContext context) { showSplashAd(); return Scaffold( appBar: AppBar( title: Text("Splash"), ), body: Center( ), ); } // 开屏 // 显示开屏广告请保证当时app内没有其他地方显示开屏广告,否则会有冲突 void showSplashAd() { if(_splashAd != null) { return; } // posId:广告位id, // imgName:背景名称, // imgLogoName:底部logo名称, // isRepeatApplyPermission:用户拒绝权限后是否允许重复读取:true允许,false禁止, // isApplyPermission:动态申请权限:true允许,false禁止, _splashAd = ADSuyiSplashAd(posId: "posid", imgName: "splash_placeholder", imgLogoName: "splash_bottom_icon", isRepeatApplyPermission: false, isApplyPermission: false); _splashAd.onClosed = (AdInfo adInfo) { print("开屏广告关闭了"); // 在加载失败和关闭回调后关闭广告 releaseSplashAd(); }; _splashAd.onFailed = (AdErrorInfo adErrorInfo) { print("开屏广告失败了"); // 在加载失败和关闭回调后关闭广告 releaseSplashAd(); }; _splashAd.onExposed = (AdInfo adInfo) { print("开屏广告曝光了"); }; _splashAd.onSucced = (AdInfo adInfo) { print("开屏广告成功了"); }; _splashAd.onClicked = (AdInfo adInfo) { print("开屏广告点击了"); }; _splashAd.loadAndShow(); } void releaseSplashAd() { _splashAd?.release(); _splashAd = null; } @override void dispose() { releaseSplashAd(); super.dispose(); } } ``` ### 4.2.2 开屏广告(flutter原生加载展示分离) ```dart class SplashLoadShowSeparatePage extends StatefulWidget { @override State createState() => _SplashLoadShowSeparateState(); } class _SplashLoadShowSeparateState extends State { ADSuyiSplashAdLoadShowSeparate? _adSuyiFlutterSplashAd; bool _hasInitBanner = false; @override Widget build(BuildContext context) { if (_adSuyiFlutterSplashAd == null && _hasInitBanner == false) { MediaQueryData queryData = MediaQuery.of(context); _hasInitBanner = true; var width = queryData.size.width; var height = queryData.size.height; _adSuyiFlutterSplashAd = ADSuyiSplashAdLoadShowSeparate( posId: KeyManager.splashPosid(), width: width, height: height); // 加载广告 _adSuyiFlutterSplashAd!.load(); _adSuyiFlutterSplashAd!.onSucced = (AdInfo adInfo) { print("开屏广告加载成功"); // 展示广告 _adSuyiFlutterSplashAd!.show(); }; _adSuyiFlutterSplashAd!.onFailed = (AdErrorInfo adErrorInfo) { releaseSplashAd(); print("开屏广告加载失败"); toHome(); }; _adSuyiFlutterSplashAd!.onClicked = (AdInfo adInfo) { print("开屏广告点击"); }; _adSuyiFlutterSplashAd!.onExposed = (AdInfo adInfo) { print("开屏广告渲染成功"); }; _adSuyiFlutterSplashAd!.onClosed = (AdInfo adInfo) { releaseSplashAd(); print("开屏广告关闭成功"); toHome(); }; } return Scaffold( body: Center( child: (_adSuyiFlutterSplashAd == null ? Text("开屏广告已关闭") : ADSuyiWidget(adView: _adSuyiFlutterSplashAd!)))); } void toHome() { Navigator.pushAndRemoveUntil( context, new MaterialPageRoute(builder: (context) => new MyApp()), (route) => route == null, ); } void removeSplashAd() { setState(() { releaseSplashAd(); }); } void releaseSplashAd() { _adSuyiFlutterSplashAd?.release(); _adSuyiFlutterSplashAd = null; } @override void dispose() { releaseSplashAd(); super.dispose(); } } ``` ```dart // 加载广告 _adSuyiFlutterSplashAd!.load(); ``` ```dart // 展示广告(请在onSucced回调后使用,目前安卓端快手存在异常,请勿添加快手广告源) _adSuyiFlutterSplashAd!.show(); ``` ## 4.3 横幅广告(banner) ```dart class BannerPage extends StatefulWidget { @override State createState() => _BannerState(); } class _BannerState extends State { ADSuyiFlutterBannerAd _adSuyiFlutterBannerAd; bool _hasInitBanner = false; @override Widget build(BuildContext context) { if (_adSuyiFlutterBannerAd == null && _hasInitBanner == false) { MediaQueryData queryData = MediaQuery.of(context); _hasInitBanner = true; var width = queryData.size.width; var height = queryData.size.width / 320.0 * 50.0; _adSuyiFlutterBannerAd = ADSuyiFlutterBannerAd(posId: KeyManager.bannerPosid(), width: width, height: height); _adSuyiFlutterBannerAd.loadAndShow(); _adSuyiFlutterBannerAd.onSucced = (AdInfo adInfo) { print("横幅广告加载成功"); }; _adSuyiFlutterBannerAd.onFailed = (AdErrorInfo adErrorInfo) { removeBannerAd(); print("横幅广告加载失败"); }; _adSuyiFlutterBannerAd.onClicked = (AdInfo adInfo) { print("横幅广告点击"); }; _adSuyiFlutterBannerAd.onExposed = (AdInfo adInfo) { print("横幅广告渲染成功"); }; _adSuyiFlutterBannerAd.onClosed = (AdInfo adInfo) { removeBannerAd(); print("横幅广告关闭成功"); }; } return Scaffold( appBar: AppBar( title: Text("BannerPage"), ), body: Center(child: (_adSuyiFlutterBannerAd == null ? Text("banner广告已关闭") : ADSuyiWidget(adView: _adSuyiFlutterBannerAd) )) ); } void removeBannerAd() { setState(() { releaseBannerAd(); }); } void releaseBannerAd() { _adSuyiFlutterBannerAd?.release(); _adSuyiFlutterBannerAd = null; } @override void dispose() { releaseBannerAd(); super.dispose(); } } ``` ## 4.4 全屏视频广告 ```dart class FullScreenPage extends StatefulWidget { @override State createState() => FullScreenState(); } class FullScreenState extends State { ADSuyiFullScreenVodAd _fullScreenVodAd; @override Widget build(BuildContext context) { showFullScreenVodAd(); return Scaffold( appBar: AppBar( title: Text("FullScreenVodAd"), ), body: Center( ), ); } // 全屏视频 // 显示全屏视频广告请保证当时app内没有其他地方显示全屏视频广告,否则会有冲突 void showFullScreenVodAd() { if(_fullScreenVodAd != null) { return; } _fullScreenVodAd = ADSuyiFullScreenVodAd(posId: KeyManager.fullScreenPosid()); _fullScreenVodAd.onClicked = (AdInfo adInfo) { print("全屏视频广告关闭了"); }; _fullScreenVodAd.onFailed = (AdErrorInfo adErrorInfo) { print("全屏视频广告失败了"); releaseFullScreenVodAd(); }; _fullScreenVodAd.onExposed = (AdInfo adInfo) { print("全屏视频广告曝光了"); }; _fullScreenVodAd.onSucced = (AdInfo adInfo) { print("全屏视频广告成功了"); playFullScreenVodAd(); }; _fullScreenVodAd.onClicked = (AdInfo adInfo) { print("全屏视频广告点击了"); }; _fullScreenVodAd.onRewarded = () { print("全屏视频广告激励达成"); }; _fullScreenVodAd.onClosed = (AdInfo adInfo) { print("全屏视频广告关闭"); releaseFullScreenVodAd(); }; _fullScreenVodAd.load(); } void releaseFullScreenVodAd() { _fullScreenVodAd?.release(); _fullScreenVodAd = null; } void playFullScreenVodAd() { _fullScreenVodAd.show(); } @override void dispose() { releaseFullScreenVodAd(); super.dispose(); } } ``` ## 4.5 插屏广告 ```dart class InterPage extends StatefulWidget { @override State createState() => _InterState(); } class _InterState extends State { ADSuyiIntertitialAd _interAd; @override Widget build(BuildContext context) { showInterAd(); return Scaffold( appBar: AppBar( title: Text("Intertitial"), ), body: Center( ), ); } // 插屏 // 显示插屏广告请保证当时app内没有其他地方显示插屏广告,否则会有冲突 void showInterAd() { if(_interAd != null) { return; } _interAd = ADSuyiIntertitialAd(posId: KeyManager.interPosid()); _interAd.onClicked = (AdInfo adInfo) { print("插屏广告关闭了"); }; _interAd.onFailed = (AdErrorInfo adErrorInfo) { print("插屏广告失败了"); releaseInterAd(); }; _interAd.onExposed = (AdInfo adInfo) { print("插屏广告曝光了"); }; _interAd.onSucced = (AdInfo adInfo) { print("插屏广告成功了"); playInterAd(); }; _interAd.onClicked = (AdInfo adInfo) { print("插屏广告点击了"); }; _interAd.onRewarded = () { print("插屏广告激励达成"); }; _interAd.onClosed = (AdInfo adInfo) { print("插屏广告关闭"); releaseInterAd(); }; _interAd.load(); } void releaseInterAd() { _interAd?.release(); _interAd = null; } void playInterAd() { _interAd.show(); } @override void dispose() { releaseInterAd(); super.dispose(); } } ``` ## 4.6 信息流广告 ```dart class NativePage extends StatefulWidget { @override State createState() => NativeState(); } class NativeState extends State { ADSuyiFlutterNativeAd _nativeAd; List _items = List.generate(10, (i) => i); ScrollController _scrollController = ScrollController(); @override void initState() { super.initState(); _scrollController.addListener(() { if (_scrollController.position.pixels == _scrollController.position.maxScrollExtent) { _getAdData(); } }); } _getAdData() async { _nativeAd.load(); } void createNativeAd(BuildContext context) { if(_nativeAd == null) { MediaQueryData queryData = MediaQuery.of(context); var width = queryData.size.width; /** * 由于flutter对接安卓原生广告无法提前获取到部分渠道广告视图的高度,开发者需要做一下处理。 * 例如每个渠道都申请大小相近的广告样式,可在此设置固定比例。(优量汇:选横版纯图片;穿山甲:文字浮层;百度:三图;快手:横版大图) * 由于demo上用的广告位素材样式不固定,我们直接写16:12进行测试。 * 设置比例的作用,主要是防止广告素材被遮挡,导致收益降低和合规问题。 */ var height = width / 16 * 12; _nativeAd = ADSuyiFlutterNativeAd(posId: KeyManager.nativePosid(), width: width, height: height); _nativeAd!.onFailed = (AdErrorInfo adErrorInfo) { }; _nativeAd.onReceived = (ADSuyiFlutterNativeAdView adView) { setState(() { var adWidget = ADSuyiWidget(adView: adView); adView.onClosed = (AdInfo adInfo) { // 广告关闭回调 setState(() { _items.remove(adWidget); adView.release(); }); }; adView.onExposed = (AdInfo adInfo) { // 广告曝光回调 }; adView.onClicked = (AdInfo adInfo) { // 广告点击回调 }; _items.add(adWidget); _items.addAll(List.generate(1, (i) => i)); }); }; } } @override Widget build(BuildContext context) { createNativeAd(context); return Scaffold( appBar: AppBar( title: Text("Native"), ), body: Center( child: ListView.builder( itemCount: _items.length, controller: _scrollController, itemBuilder: (BuildContext context, int index) { final item = _items[index]; if (item is Widget) { return item; } else { return Container( width: 300, height: 150, child: Text("Cell", style: TextStyle(fontSize: 75)) ); } } ), ), ); } @override void dispose() { for (var item in _items) { if (item is ADSuyiFlutterNativeAdView) { item.release(); } } _nativeAd.release(); _nativeAd = null; super.dispose(); } } ``` ## 4.7 激励视频广告 ```dart class RewardPage extends StatefulWidget { @override State createState() => _RewardState(); } class _RewardState extends State { ADSuyiRewardAd _rewardAd; @override Widget build(BuildContext context) { showRewardAd(); return Scaffold( appBar: AppBar( title: Text("Reward"), ), body: Center( ), ); } void showRewardAd() { if(_rewardAd != null) { return; } _rewardAd = ADSuyiRewardAd(posId: KeyManager.rewardPosid()); _rewardAd.onClicked = (AdInfo adInfo) { print("激励视频广告关闭了"); }; _rewardAd.onFailed = (AdErrorInfo adErrorInfo) { print("激励视频广告失败了"); releaseRewardAd(); }; _rewardAd.onExposed = (AdInfo adInfo) { print("激励视频广告曝光了"); }; _rewardAd.onSucced = (AdInfo adInfo) { print("激励视频广告成功了"); playRewardAd(); }; _rewardAd.onClicked = (AdInfo adInfo) { print("激励视频广告点击了"); }; _rewardAd.onRewarded = (AdInfo adInfo) { print("激励视频广告激励达成"); }; _rewardAd.onClosed = (AdInfo adInfo) { print("激励视频广告关闭"); releaseRewardAd(); }; _rewardAd.load(); } void releaseRewardAd() { _rewardAd?.release(); _rewardAd = null; } void playRewardAd() { _rewardAd.show(); } @override void dispose() { releaseRewardAd(); super.dispose(); } } ``` ## 4.8 个性化化开关 ```dart AdSuyiFlutterSdk.setPersonalizedEnabled(personalized: true); ``` ## 4.9 获取广告信息 目前onSucced、onClosed、onClicked、onExposed、onRewarded回调增加adInfo回传,开发者可通过以下方法获取广告信息 | 方法名 | 介绍 | | ------------ | ---- | | platformName | 获取三方广告平台名称,返回String类型。| | ecpm | 获取ECPM,返回Double类型(单位:元)。| | ecpmPrecision | ECPM类型,返回String类型(accurate:精准、platform_assignment:平台指定、estimate:估算)。| ## 4.5 获取广告错误信息 目前仅onFailed方法会回调,开发者可通过以下方法获取广告错误信息 | 方法名 | 介绍 | | ------------ | ---- | | errorCode | 错误码,返回int类型。| | errorDescription | 获取错误信息,返回String类型。|