Commit ce276060 authored by wanglei's avatar wanglei

[优化]添加插页场景

parent e2effc51
...@@ -169,19 +169,26 @@ dependencies { ...@@ -169,19 +169,26 @@ dependencies {
//广告 //广告
//admob渠道 //admob渠道
implementation(libs.vungle) // implementation(libs.vungle)
implementation(libs.facebook) // implementation(libs.facebook)
implementation(libs.mintegral) // implementation(libs.mintegral)
implementation(libs.pangle) // implementation(libs.pangle)
//applovin sdk //applovin sdk
implementation(libs.applovin) // implementation(libs.applovin)
//applovin渠道 //applovin渠道
implementation(libs.applovin.google) // implementation(libs.applovin.google)
implementation(libs.applovin.admob) // implementation(libs.applovin.admob)
implementation(libs.applovin.facebook) //meta // implementation(libs.applovin.facebook) //meta
implementation(libs.applovin.mintegral)//mintegral // implementation(libs.applovin.mintegral)//mintegral
implementation(libs.applovin.pangle) //pangle // implementation(libs.applovin.pangle) //pangle
implementation(libs.applovin.vungle) //vungle // implementation(libs.applovin.vungle) //vungle
implementation("com.google.android.gms:play-services-ads:23.5.0")
implementation("com.google.ads.mediation:applovin:13.0.1.0")
implementation("com.google.ads.mediation:facebook:6.18.0.0")
implementation("com.google.ads.mediation:mintegral:16.8.61.0")
implementation("com.google.ads.mediation:pangle:6.3.0.4.0")
implementation("com.google.ads.mediation:vungle:7.4.2.0")
val work_version = "2.8.1" val work_version = "2.8.1"
implementation("androidx.work:work-runtime-ktx:$work_version") implementation("androidx.work:work-runtime-ktx:$work_version")
...@@ -190,5 +197,7 @@ dependencies { ...@@ -190,5 +197,7 @@ dependencies {
implementation("com.android.billingclient:billing:$billing_version") implementation("com.android.billingclient:billing:$billing_version")
implementation("com.android.billingclient:billing-ktx:$billing_version") implementation("com.android.billingclient:billing-ktx:$billing_version")
implementation("com.github.JavaNoober.BackgroundLibrary:libraryx:1.7.6")
} }
\ No newline at end of file
...@@ -46,6 +46,12 @@ ...@@ -46,6 +46,12 @@
android:exported="false" android:exported="false"
android:screenOrientation="portrait" android:screenOrientation="portrait"
tools:ignore="DiscouragedApi,LockedOrientationActivity" /> tools:ignore="DiscouragedApi,LockedOrientationActivity" />
<activity
android:name=".ui.guide.CleanGuideActivity"
android:exported="false"
android:screenOrientation="portrait"
tools:ignore="DiscouragedApi,LockedOrientationActivity" />
<activity <activity
android:name=".ui.guide.GuideActivity" android:name=".ui.guide.GuideActivity"
android:exported="false" android:exported="false"
......
...@@ -4,14 +4,16 @@ package com.dumpster.cleaner.bean.config ...@@ -4,14 +4,16 @@ package com.dumpster.cleaner.bean.config
class AdConfigBean( class AdConfigBean(
var isAdShow: Boolean = true,//广告开关 var isAdShow: Boolean = true,//广告开关
var adSwitch: Boolean = true,//true 走admob,false走max
var taichiAdValue: Int = 1,//价值上报阀值 var taichiAdValue: Int = 1,//价值上报阀值
var adRatio: Int = 100,//价值上报随机控制 var adRatio: Int = 100,//价值上报随机控制
var numDisplayLimit: Int = -1,//展示次数限制 var numDisplayLimit: Int = -1,//展示次数限制
var numRequestLimit: Int = -1,//请求次数限制 var numRequestLimit: Int = -1,//请求次数限制
var numClickLimit: Int = -1,//点击次数限制 var numClickLimit: Int = -1,//点击次数限制
var timeInterval: Int = 10,//广告间隔秒
var timeInterval: Int = 10,//插页广告间隔秒
var timeIntervalOpen: Int = 0,//开屏广告间隔
var openAdLoading: Int = 15,//开屏广告拉取时间 var openAdLoading: Int = 15,//开屏广告拉取时间
var numNativeDisplayLimit: Int = -1,//原生展示次数限制 var numNativeDisplayLimit: Int = -1,//原生展示次数限制
......
...@@ -6,9 +6,9 @@ package com.dumpster.cleaner.bean.config ...@@ -6,9 +6,9 @@ package com.dumpster.cleaner.bean.config
*/ */
data class ConfigBean( data class ConfigBean(
var isInBlackList: Boolean = false, // var isInBlackList: Boolean = false,
val ut: Int = 0, val ut: Int = 0,
var vpnCanUse: Boolean = true, // var vpnCanUse: Boolean = true,
val adConfigBean: AdConfigBean = AdConfigBean(), val adConfigBean: AdConfigBean = AdConfigBean(),
val popupConfigBean: PopupConfigBean = PopupConfigBean(), val popupConfigBean: PopupConfigBean = PopupConfigBean(),
val vipConfigBean: VipConfigBean = VipConfigBean(), val vipConfigBean: VipConfigBean = VipConfigBean(),
......
...@@ -41,9 +41,9 @@ abstract class AdEvent { ...@@ -41,9 +41,9 @@ abstract class AdEvent {
fun adPulStart() { fun adPulStart() {
val obj = JSONObject() val obj = JSONObject()
obj.put("req_id", reqId)
obj.put("from", from)
obj.put("ad_unit", adUnit) obj.put("ad_unit", adUnit)
obj.put("from", from)
obj.put("req_id", reqId)
obj.put("ad_type", adUnit) obj.put("ad_type", adUnit)
EventUtils.event("ad_pull_start", ext = obj) EventUtils.event("ad_pull_start", ext = obj)
LogEx.logDebug(TAG, "ad_pull_start_$adUnit $obj") LogEx.logDebug(TAG, "ad_pull_start_$adUnit $obj")
......
package com.dumpster.cleaner.business.ads package com.dumpster.cleaner.business.ads
import android.app.Dialog import android.app.Dialog
import com.dumpster.cleaner.business.ads.LimitUtils.openInterLastShowTime import com.dumpster.cleaner.business.ads.AdsType.Companion.INSERT
import com.dumpster.cleaner.business.ads.AdsType.Companion.OPEN
import com.dumpster.cleaner.business.ads.LimitUtils.interLastShowTime
import com.dumpster.cleaner.business.ads.LimitUtils.openLastShowTime
class AdState<T>() { class AdState<T>() {
...@@ -34,20 +37,32 @@ class AdState<T>() { ...@@ -34,20 +37,32 @@ class AdState<T>() {
/** /**
* 广告已经展示 * 广告已经展示
*/ */
fun onAdDisplayed() { fun onAdDisplayed(adsType: AdsType) {
currentAd = null currentAd = null
currentAdEvent = null currentAdEvent = null
adDialog?.dismiss() adDialog?.dismiss()
adDialog = null adDialog = null
openInterLastShowTime = System.currentTimeMillis() if (adsType == OPEN) {
openLastShowTime = System.currentTimeMillis()
}
if (adsType == INSERT) {
interLastShowTime = System.currentTimeMillis()
}
} }
fun onAdHidden() { fun onAdHidden(adsType: AdsType) {
//重置下上次展示的时间,避免看广告的时间算入间隔 //重置下上次展示的时间,避免看广告的时间算入间隔
openInterLastShowTime = System.currentTimeMillis() if (adsType == OPEN) {
openLastShowTime = System.currentTimeMillis()
}
if (adsType == INSERT) {
interLastShowTime = System.currentTimeMillis()
}
} }
...@@ -59,14 +74,14 @@ class AdState<T>() { ...@@ -59,14 +74,14 @@ class AdState<T>() {
currentAdEvent = null currentAdEvent = null
} }
fun onAdLoaded(ad: T?, adEvent: AdEvent?) { fun loadStart(adEvent: AdEvent) {
//这里可能提前设置,所有可以不设置,max回调的类型可能不同 loadingAd = true
if (ad != null) { currentAdEvent = adEvent
currentAd = ad }
}
if (adEvent != null) { fun onAdLoaded(ad: T?) {
currentAdEvent = adEvent currentAd = ad
}
loadingAd = false loadingAd = false
lastLoadTime = System.currentTimeMillis() lastLoadTime = System.currentTimeMillis()
} }
...@@ -81,4 +96,5 @@ class AdState<T>() { ...@@ -81,4 +96,5 @@ class AdState<T>() {
fun adAvailable() = fun adAvailable() =
currentAd != null || ((System.currentTimeMillis() - lastLoadTime) / 1000 / 60).toInt() < 30 currentAd != null || ((System.currentTimeMillis() - lastLoadTime) / 1000 / 60).toInt() < 30
} }
...@@ -18,10 +18,10 @@ import com.dumpster.cleaner.business.ads.admob.AdInterMgr ...@@ -18,10 +18,10 @@ import com.dumpster.cleaner.business.ads.admob.AdInterMgr
import com.dumpster.cleaner.business.ads.admob.AdNativeMgr import com.dumpster.cleaner.business.ads.admob.AdNativeMgr
import com.dumpster.cleaner.business.ads.admob.AdOpenMgr import com.dumpster.cleaner.business.ads.admob.AdOpenMgr
import com.dumpster.cleaner.business.ads.admob.AdmobEvent import com.dumpster.cleaner.business.ads.admob.AdmobEvent
import com.dumpster.cleaner.business.ads.applovin.AdMaxEvent //import com.dumpster.cleaner.business.ads.applovin.AdMaxEvent
import com.dumpster.cleaner.business.ads.applovin.MaxInsertMgr //import com.dumpster.cleaner.business.ads.applovin.MaxInsertMgr
import com.dumpster.cleaner.business.ads.applovin.MaxNativeMgr //import com.dumpster.cleaner.business.ads.applovin.MaxNativeMgr
import com.dumpster.cleaner.business.ads.applovin.MaxOpenMgr //import com.dumpster.cleaner.business.ads.applovin.MaxOpenMgr
import com.dumpster.cleaner.business.helper.EventUtils import com.dumpster.cleaner.business.helper.EventUtils
import com.dumpster.cleaner.utils.AppPreferences import com.dumpster.cleaner.utils.AppPreferences
import com.dumpster.cleaner.utils.LogEx import com.dumpster.cleaner.utils.LogEx
...@@ -49,15 +49,15 @@ object AdsMgr { ...@@ -49,15 +49,15 @@ object AdsMgr {
private val adBannerMgr by lazy { private val adBannerMgr by lazy {
AdBannerMgr() AdBannerMgr()
} }
private val maxOpenMgr by lazy { // private val maxOpenMgr by lazy {
MaxOpenMgr() // MaxOpenMgr()
} // }
private val maxInsertMgr by lazy { // private val maxInsertMgr by lazy {
MaxInsertMgr() // MaxInsertMgr()
} // }
private val maxNativeMgr by lazy { // private val maxNativeMgr by lazy {
MaxNativeMgr() // MaxNativeMgr()
} // }
/** /**
* 是否初始化 * 是否初始化
...@@ -87,12 +87,12 @@ object AdsMgr { ...@@ -87,12 +87,12 @@ object AdsMgr {
*/ */
fun init(context: Context) { fun init(context: Context) {
if (configBean.isInBlackList) { // if (configBean.isInBlackList) {
EventUtils.event("isInBlackList", value = "isInBlackList=${configBean.isInBlackList}") // EventUtils.event("isInBlackList", value = "isInBlackList=${configBean.isInBlackList}")
return // return
} // }
initAdmob(context) initAdmob(context)
initMax(context) // initMax(context)
} }
...@@ -106,7 +106,7 @@ object AdsMgr { ...@@ -106,7 +106,7 @@ object AdsMgr {
EventUtils.event("AdmobInit", "AdmobInit=$isAdmobInit") EventUtils.event("AdmobInit", "AdmobInit=$isAdmobInit")
// context.toast("admob init") // context.toast("admob init")
if (adsConfigBean.adSwitch) { if (true) {
admobInitCallBack?.invoke() admobInitCallBack?.invoke()
admobInitCallBack = null admobInitCallBack = null
adNativeMgr.loadAd(context, AdmobEvent("nativeAd", context::class.java.simpleName)) adNativeMgr.loadAd(context, AdmobEvent("nativeAd", context::class.java.simpleName))
...@@ -117,38 +117,38 @@ object AdsMgr { ...@@ -117,38 +117,38 @@ object AdsMgr {
} }
private fun initMax(context: Context) = kotlin.runCatching { // private fun initMax(context: Context) = kotlin.runCatching {
if (isMaxInit) return@runCatching // if (isMaxInit) return@runCatching
val executor = Executors.newSingleThreadExecutor() // val executor = Executors.newSingleThreadExecutor()
//
executor.execute { // executor.execute {
val currentGaid = AdvertisingIdClient.getAdvertisingIdInfo(context).id // val currentGaid = AdvertisingIdClient.getAdvertisingIdInfo(context).id
AppPreferences.getInstance().getString("gid", currentGaid) // AppPreferences.getInstance().getString("gid", currentGaid)
//
val build = AppLovinSdkInitializationConfiguration // val build = AppLovinSdkInitializationConfiguration
.builder(GlobalConfig.KEY_MAX, context) // .builder(GlobalConfig.KEY_MAX, context)
//
build.mediationProvider = AppLovinMediationProvider.MAX // build.mediationProvider = AppLovinMediationProvider.MAX
if (BuildConfig.DEBUG) { // if (BuildConfig.DEBUG) {
build.testDeviceAdvertisingIds = Collections.singletonList(currentGaid) // build.testDeviceAdvertisingIds = Collections.singletonList(currentGaid)
} // }
val initConfig = build.build() // val initConfig = build.build()
runCatching { // runCatching {
AppLovinSdk.getInstance(context).initialize(initConfig) { // AppLovinSdk.getInstance(context).initialize(initConfig) {
isMaxInit = true // isMaxInit = true
// maxOpenMgr.loadAd(context) //// maxOpenMgr.loadAd(context)
if (!adsConfigBean.adSwitch) { // if (!adsConfigBean.adSwitch) {
// maxInsertMgr.loadAd(context, AdMaxEvent("interAd", context::class.java.simpleName)) //// maxInsertMgr.loadAd(context, AdMaxEvent("interAd", context::class.java.simpleName))
context.toast("max init") // context.toast("max init")
maxInitCallBack?.invoke() // maxInitCallBack?.invoke()
maxInitCallBack = null // maxInitCallBack = null
} // }
} // }
} // }
//
} // }
//
} // }
var admobInitCallBack: (() -> Unit)? = null var admobInitCallBack: (() -> Unit)? = null
var maxInitCallBack: (() -> Unit)? = null var maxInitCallBack: (() -> Unit)? = null
...@@ -172,14 +172,14 @@ object AdsMgr { ...@@ -172,14 +172,14 @@ object AdsMgr {
return return
} }
if (configBean.isInBlackList) { // if (configBean.isInBlackList) {
EventUtils.event("isInBlackList", "isInBlackList=${configBean.isInBlackList}") // EventUtils.event("isInBlackList", "isInBlackList=${configBean.isInBlackList}")
showCallBack?.failed() // showCallBack?.failed()
return // return
} // }
val from = activity::class.java.simpleName val from = activity::class.java.simpleName
if (adsConfigBean.adSwitch) { if (true) {
val admobEvent = AdmobEvent("openAd", from).apply { this.isUnLimit = isUnLimit } val admobEvent = AdmobEvent("openAd", from).apply { this.isUnLimit = isUnLimit }
if (isAdmobInit) { if (isAdmobInit) {
adOpenMgr.show(activity, admobEvent, showCallBack) adOpenMgr.show(activity, admobEvent, showCallBack)
...@@ -189,13 +189,13 @@ object AdsMgr { ...@@ -189,13 +189,13 @@ object AdsMgr {
adOpenMgr.show(activity, admobEvent, showCallBack) adOpenMgr.show(activity, admobEvent, showCallBack)
} }
} else { } else {
if (isMaxInit) { // if (isMaxInit) {
maxOpenMgr.show(activity, isUnLimit, AdMaxEvent("openAd", from), showCallBack) // maxOpenMgr.show(activity, isUnLimit, AdMaxEvent("openAd", from), showCallBack)
} else { // } else {
maxInitCallBack = { // maxInitCallBack = {
maxOpenMgr.show(activity, isUnLimit, AdMaxEvent("openAd", from), showCallBack) // maxOpenMgr.show(activity, isUnLimit, AdMaxEvent("openAd", from), showCallBack)
} // }
} // }
} }
} }
...@@ -219,17 +219,16 @@ object AdsMgr { ...@@ -219,17 +219,16 @@ object AdsMgr {
return return
} }
if (configBean.isInBlackList) { // if (configBean.isInBlackList) {
EventUtils.event("isInBlackList", configBean.isInBlackList.toString()) // EventUtils.event("isInBlackList", configBean.isInBlackList.toString())
showCallBack?.failed() // showCallBack?.failed()
return // return
} // }
LogEx.logDebug("showAd", "adSwitch=${adsConfigBean.adSwitch}")
val from = activity::class.java.simpleName val from = activity::class.java.simpleName
if (adsConfigBean.adSwitch) { if (true) {
adInterMgr.show(activity, AdmobEvent("interAd", from).apply { this.isUnLimit = isUnLimit }, showCallBack) adInterMgr.show(activity, AdmobEvent("interAd", from).apply { this.isUnLimit = isUnLimit }, showCallBack)
} else { } else {
maxInsertMgr.show(activity, isUnLimit, AdMaxEvent("interAd", from), showCallBack) // maxInsertMgr.show(activity, isUnLimit, AdMaxEvent("interAd", from), showCallBack)
} }
} }
...@@ -250,16 +249,16 @@ object AdsMgr { ...@@ -250,16 +249,16 @@ object AdsMgr {
return return
} }
nativeView.visibility = View.VISIBLE nativeView.visibility = View.VISIBLE
if (configBean.isInBlackList) { // if (configBean.isInBlackList) {
EventUtils.event("isInBlackList", configBean.isInBlackList.toString()) // EventUtils.event("isInBlackList", configBean.isInBlackList.toString())
return // return
} // }
val showNative = { val showNative = {
if (adsConfigBean.adSwitch) { if (true) {
adNativeMgr.show(AdmobEvent("nativeAd", "nativeAd"), nativeView, layout, nativeCallBack) adNativeMgr.show(AdmobEvent("nativeAd", "nativeAd"), nativeView, layout, nativeCallBack)
} else { } else {
maxNativeMgr.show(AdMaxEvent("nativeAd", "nativeAd"), nativeView, layout, nativeCallBack) // maxNativeMgr.show(AdMaxEvent("nativeAd", "nativeAd"), nativeView, layout, nativeCallBack)
} }
} }
...@@ -282,11 +281,11 @@ object AdsMgr { ...@@ -282,11 +281,11 @@ object AdsMgr {
return return
} }
parent.visibility = View.VISIBLE parent.visibility = View.VISIBLE
if (configBean.isInBlackList) { // if (configBean.isInBlackList) {
EventUtils.event("isInBlackList", configBean.isInBlackList.toString()) // EventUtils.event("isInBlackList", configBean.isInBlackList.toString())
return // return
} // }
if (adsConfigBean.adSwitch) { if (true) {
adBannerMgr.show(parent, collapsible, adClose) adBannerMgr.show(parent, collapsible, adClose)
} }
} }
......
...@@ -151,26 +151,48 @@ object LimitUtils { ...@@ -151,26 +151,48 @@ object LimitUtils {
/** /**
* 开屏和插页广告的显示间隔限制 * 开屏限制
*/ */
fun isIntervalLimited(adEvent: AdEvent): Boolean { fun isIntervalOpenLimit(adEvent: AdEvent): Boolean {
val flag = ((System.currentTimeMillis() - openInterLastShowTime) / 1000).toInt() < (AdConfigBean.adsConfigBean.timeInterval) val flag = ((System.currentTimeMillis() - openLastShowTime) / 1000).toInt() < (AdConfigBean.adsConfigBean.timeIntervalOpen)
if (flag) { if (flag) {
adEvent.adShowError("ad in timeInterval") adEvent.adShowError("ad in timeInterval")
} }
return flag return flag
} }
//开屏和插页上一次展示时间共用,避免开屏插页连弹 /**
var openInterLastShowTime = 0L * 插屏限制
*/
fun isIntervalInterLimit(adEvent: AdEvent): Boolean {
val flag = ((System.currentTimeMillis() - interLastShowTime) / 1000).toInt() < (AdConfigBean.adsConfigBean.timeInterval)
if (flag) {
adEvent.adShowError("ad in timeInterval")
}
return flag
}
//开屏上次展示时间
var openLastShowTime = 0L
get() { get() {
return AppPreferences.getInstance().getLong("openInterLastShowTime", field) return AppPreferences.getInstance().getLong("openLastShowTime", field)
} }
set(value) { set(value) {
field = value field = value
AppPreferences.getInstance().put("openInterLastShowTime", value, true) AppPreferences.getInstance().put("openLastShowTime", value, true)
} }
//插屏上次展示时间
var interLastShowTime = 0L
get() {
return AppPreferences.getInstance().getLong("interLastShowTime", field)
}
set(value) {
field = value
AppPreferences.getInstance().put("interLastShowTime", value, true)
}
/** /**
* 原生广告是否到达限制 * 原生广告是否到达限制
*/ */
......
...@@ -4,7 +4,6 @@ import android.os.Bundle ...@@ -4,7 +4,6 @@ import android.os.Bundle
import android.view.ViewGroup import android.view.ViewGroup
import android.view.ViewTreeObserver import android.view.ViewTreeObserver
import com.dumpster.cleaner.GlobalConfig import com.dumpster.cleaner.GlobalConfig
import com.dumpster.cleaner.bean.config.AdConfigBean
import com.dumpster.cleaner.business.ads.AdsType import com.dumpster.cleaner.business.ads.AdsType
import com.dumpster.cleaner.business.ads.LimitUtils import com.dumpster.cleaner.business.ads.LimitUtils
import com.dumpster.cleaner.business.ads.admob.AdmobEvent.AdmobOnPaidEventListener import com.dumpster.cleaner.business.ads.admob.AdmobEvent.AdmobOnPaidEventListener
...@@ -26,9 +25,9 @@ class AdBannerMgr { ...@@ -26,9 +25,9 @@ class AdBannerMgr {
fun show(parent: ViewGroup, collapsible: Boolean, adClose: (() -> Unit)? = null) { fun show(parent: ViewGroup, collapsible: Boolean, adClose: (() -> Unit)? = null) {
if (!AdConfigBean.adsConfigBean.adSwitch) { // if (!AdConfigBean.adsConfigBean.adSwitch) {
return // return
} // }
val admobEvent = AdmobEvent("banner", "banner") val admobEvent = AdmobEvent("banner", "banner")
if (!LimitUtils.isAdShow(AdsType.BANNER, admobEvent)) { if (!LimitUtils.isAdShow(AdsType.BANNER, admobEvent)) {
......
...@@ -11,6 +11,7 @@ import com.dumpster.cleaner.business.ads.AdEvent ...@@ -11,6 +11,7 @@ import com.dumpster.cleaner.business.ads.AdEvent
import com.dumpster.cleaner.business.ads.AdState import com.dumpster.cleaner.business.ads.AdState
import com.dumpster.cleaner.business.ads.AdsShowCallBack import com.dumpster.cleaner.business.ads.AdsShowCallBack
import com.dumpster.cleaner.business.ads.AdsType import com.dumpster.cleaner.business.ads.AdsType
import com.dumpster.cleaner.business.ads.AdsType.Companion.OPEN
import com.dumpster.cleaner.business.ads.LimitUtils import com.dumpster.cleaner.business.ads.LimitUtils
import com.dumpster.cleaner.business.ads.admob.AdmobEvent.AdmobOnPaidEventListener import com.dumpster.cleaner.business.ads.admob.AdmobEvent.AdmobOnPaidEventListener
import com.dumpster.cleaner.utils.LogEx import com.dumpster.cleaner.utils.LogEx
...@@ -32,7 +33,7 @@ class AdInterMgr { ...@@ -32,7 +33,7 @@ class AdInterMgr {
//正在加载回调 //正在加载回调
private var loadingCallBack: (() -> Unit)? = null private var loadingCallBack: ((flag: Boolean) -> Unit)? = null
fun show( fun show(
activity: Activity, activity: Activity,
...@@ -44,16 +45,15 @@ class AdInterMgr { ...@@ -44,16 +45,15 @@ class AdInterMgr {
return return
} }
val nowAdEvent = adEvent
//currentAdEvent!=null 表示有缓存广告,关联reqId //currentAdEvent!=null 表示有缓存广告,关联reqId
adState.currentAdEvent?.let { nowAdEvent.reqId = it.reqId } adState.currentAdEvent?.let { adEvent.reqId = it.reqId }
if (!nowAdEvent.isUnLimit) { if (!adEvent.isUnLimit) {
if (!LimitUtils.isAdShow(AdsType.INSERT, nowAdEvent)) { if (!LimitUtils.isAdShow(AdsType.INSERT, adEvent)) {
showCallBack?.failed(2) showCallBack?.failed(2)
return return
} }
if (LimitUtils.isIntervalLimited(nowAdEvent)) { if (LimitUtils.isIntervalInterLimit(adEvent)) {
showCallBack?.failed(3) showCallBack?.failed(3)
return return
} }
...@@ -69,24 +69,29 @@ class AdInterMgr { ...@@ -69,24 +69,29 @@ class AdInterMgr {
adState.adDialog?.dismiss() adState.adDialog?.dismiss()
} }
nowAdEvent.adPrepareShow() adEvent.adPrepareShow()
LogEx.logDebug(adEvent.TAG, "needLoad=$needLoad") LogEx.logDebug(adEvent.TAG, "needLoad=$needLoad")
if (needLoad) { if (needLoad) {
if (!adState.loadingAd) { if (!adState.loadingAd) {
LogEx.logDebug(adEvent.TAG, "inter adState !loadingAd") LogEx.logDebug(adEvent.TAG, "inter adState !loadingAd")
loadAd(activity, nowAdEvent) { loadAd(activity, adEvent) {
showReadyAd(activity, nowAdEvent) if (it) {
showReadyAd(activity, adEvent)
} else {
showCallBack?.adFailed()
}
} }
} else { } else {
LogEx.logDebug(adEvent.TAG, "inter adState is loadingAd") LogEx.logDebug(adEvent.TAG, "inter adState is loadingAd")
loadingCallBack = { loadingCallBack = {
showReadyAd(activity, nowAdEvent) showReadyAd(activity, adEvent)
} }
} }
} else { } else {
LogEx.logDebug(adEvent.TAG, "inter ad ready") LogEx.logDebug(adEvent.TAG, "inter ad ready")
showReadyAd(activity, nowAdEvent) showReadyAd(activity, adEvent)
} }
} }
...@@ -113,7 +118,7 @@ class AdInterMgr { ...@@ -113,7 +118,7 @@ class AdInterMgr {
admobEvent.showAd(responseInfo, ac) admobEvent.showAd(responseInfo, ac)
adState.onAdDisplayed() adState.onAdDisplayed(OPEN)
showCallBack?.show() showCallBack?.show()
LimitUtils.addDisplayNum() LimitUtils.addDisplayNum()
...@@ -132,7 +137,7 @@ class AdInterMgr { ...@@ -132,7 +137,7 @@ class AdInterMgr {
override fun onAdDismissedFullScreenContent() { override fun onAdDismissedFullScreenContent() {
super.onAdDismissedFullScreenContent() super.onAdDismissedFullScreenContent()
adState.onAdHidden() adState.onAdHidden(OPEN)
showCallBack?.close() showCallBack?.close()
showCallBack = null showCallBack = null
...@@ -160,7 +165,7 @@ class AdInterMgr { ...@@ -160,7 +165,7 @@ class AdInterMgr {
fun loadAd( fun loadAd(
context: Context, context: Context,
adEvent: AdEvent, adEvent: AdEvent,
loadCallBack: (() -> Unit)? = null loadCallBack: ((flag: Boolean) -> Unit)? = null
) { ) {
if (!adEvent.isUnLimit) { if (!adEvent.isUnLimit) {
if (!LimitUtils.isAdShow(AdsType.INSERT, adEvent)) { if (!LimitUtils.isAdShow(AdsType.INSERT, adEvent)) {
...@@ -172,14 +177,12 @@ class AdInterMgr { ...@@ -172,14 +177,12 @@ class AdInterMgr {
} }
//避免无效预加载 //避免无效预加载
if (adState.loadingAd && loadCallBack == null && loadingCallBack == null) { if (adState.loadingAd) {
//容错机制
adState.loadingAd = false
return return
} }
adState.loadingAd = true
adEvent.adPulStart() adEvent.adPulStart()
adState.loadStart(adEvent)
InterstitialAd.load( InterstitialAd.load(
context, GlobalConfig.ID_ADMOB_INTER, AdRequest.Builder().build(), context, GlobalConfig.ID_ADMOB_INTER, AdRequest.Builder().build(),
...@@ -187,10 +190,10 @@ class AdInterMgr { ...@@ -187,10 +190,10 @@ class AdInterMgr {
override fun onAdLoaded(ad: InterstitialAd) { override fun onAdLoaded(ad: InterstitialAd) {
val event = (adEvent as AdmobEvent) val event = (adEvent as AdmobEvent)
ad.onPaidEventListener = AdmobOnPaidEventListener(ad, adEvent.scope) ad.onPaidEventListener = AdmobOnPaidEventListener(ad, adEvent.scope)
adState.onAdLoaded(ad, adEvent) adState.onAdLoaded(ad)
loadCallBack?.invoke() loadCallBack?.invoke(true)
loadingCallBack?.invoke() loadingCallBack?.invoke(true)
loadingCallBack = null loadingCallBack = null
LimitUtils.addRequestNum() LimitUtils.addRequestNum()
...@@ -199,11 +202,14 @@ class AdInterMgr { ...@@ -199,11 +202,14 @@ class AdInterMgr {
override fun onAdFailedToLoad(loadAdError: LoadAdError) { override fun onAdFailedToLoad(loadAdError: LoadAdError) {
adState.onAdLoadFailed() adState.onAdLoadFailed()
if (loadCallBack != null) { if (loadCallBack != null) {
adState.onAdDisplayFailed() adState.onAdDisplayFailed()
} }
showCallBack?.adFailed() loadCallBack?.invoke(false)
showCallBack = null loadingCallBack?.invoke(false)
loadingCallBack = null
(adEvent as AdmobEvent).pullAd(loadAdError.responseInfo, loadAdError) (adEvent as AdmobEvent).pullAd(loadAdError.responseInfo, loadAdError)
} }
......
...@@ -8,6 +8,7 @@ import com.dumpster.cleaner.business.ads.AdEvent ...@@ -8,6 +8,7 @@ import com.dumpster.cleaner.business.ads.AdEvent
import com.dumpster.cleaner.business.ads.AdState import com.dumpster.cleaner.business.ads.AdState
import com.dumpster.cleaner.business.ads.AdsShowCallBack import com.dumpster.cleaner.business.ads.AdsShowCallBack
import com.dumpster.cleaner.business.ads.AdsType import com.dumpster.cleaner.business.ads.AdsType
import com.dumpster.cleaner.business.ads.AdsType.Companion.INSERT
import com.dumpster.cleaner.business.ads.LimitUtils import com.dumpster.cleaner.business.ads.LimitUtils
import com.dumpster.cleaner.business.ads.admob.AdmobEvent.AdmobOnPaidEventListener import com.dumpster.cleaner.business.ads.admob.AdmobEvent.AdmobOnPaidEventListener
import com.dumpster.cleaner.utils.LogEx import com.dumpster.cleaner.utils.LogEx
...@@ -27,7 +28,7 @@ class AdOpenMgr { ...@@ -27,7 +28,7 @@ class AdOpenMgr {
private var showCallBack: AdsShowCallBack? = null private var showCallBack: AdsShowCallBack? = null
//正在加载回调 //正在加载回调
private var loadingCallBack: (() -> Unit)? = null private var loadingCallBack: ((flag: Boolean) -> Unit)? = null
fun show( fun show(
activity: Activity, activity: Activity,
...@@ -38,15 +39,14 @@ class AdOpenMgr { ...@@ -38,15 +39,14 @@ class AdOpenMgr {
return return
} }
val nowAdEvent = adEvent adState.currentAdEvent?.let { adEvent.reqId = it.reqId }
adState.currentAdEvent?.let { nowAdEvent.reqId = it.reqId }
if (!nowAdEvent.isUnLimit) { if (!adEvent.isUnLimit) {
if (!LimitUtils.isAdShow(AdsType.OPEN, nowAdEvent)) { if (!LimitUtils.isAdShow(AdsType.OPEN, adEvent)) {
showCallBack?.failed() showCallBack?.failed()
return return
} }
if (LimitUtils.isIntervalLimited(nowAdEvent)) { if (LimitUtils.isIntervalOpenLimit(adEvent)) {
showCallBack?.failed() showCallBack?.failed()
return return
} }
...@@ -55,26 +55,36 @@ class AdOpenMgr { ...@@ -55,26 +55,36 @@ class AdOpenMgr {
val needLoad = !adState.adAvailable() val needLoad = !adState.adAvailable()
this.showCallBack = showCallBack this.showCallBack = showCallBack
nowAdEvent.adPrepareShow() adEvent.adPrepareShow()
if (needLoad) { if (needLoad) {
if (!adState.loadingAd) { if (!adState.loadingAd) {
LogEx.logDebug(adEvent.TAG, "open adState !loadingAd") LogEx.logDebug(adEvent.TAG, "open adState !loadingAd")
loadAd(activity, adEvent) { loadAd(activity, adEvent) {
showReadyAd(activity) if (it) {
showReadyAd(activity, adEvent)
} else {
showCallBack?.adFailed()
}
} }
} else { } else {
LogEx.logDebug(adEvent.TAG, "open adState is loadingAd") LogEx.logDebug(adEvent.TAG, "open adState is loadingAd")
loadingCallBack = { loadingCallBack = {
showReadyAd(activity) if (it) {
showReadyAd(activity, adEvent)
} else {
showCallBack?.adFailed()
}
} }
} }
} else { } else {
LogEx.logDebug(adEvent.TAG, "open ad ready") LogEx.logDebug(adEvent.TAG, "open ad ready")
showReadyAd(activity) showReadyAd(activity, adEvent)
} }
} }
private fun showReadyAd(ac: Activity) { private fun showReadyAd(ac: Activity, adEvent: AdEvent) {
val admobEvent = (adEvent as AdmobEvent)
if (ac.isFinishing || ac.isDestroyed || adState.currentAd == null) { if (ac.isFinishing || ac.isDestroyed || adState.currentAd == null) {
LogEx.logDebug(TAG, "showReadyAd ac=null isFinishing isDestroyed") LogEx.logDebug(TAG, "showReadyAd ac=null isFinishing isDestroyed")
...@@ -83,16 +93,15 @@ class AdOpenMgr { ...@@ -83,16 +93,15 @@ class AdOpenMgr {
adState.currentAd?.run { adState.currentAd?.run {
val adEvent = adState.currentAdEvent as AdmobEvent?
fullScreenContentCallback = object : FullScreenContentCallback() { fullScreenContentCallback = object : FullScreenContentCallback() {
override fun onAdShowedFullScreenContent() { override fun onAdShowedFullScreenContent() {
adEvent?.showAd(this@run.responseInfo, ac) admobEvent.showAd(this@run.responseInfo, ac)
showCallBack?.show() showCallBack?.show()
adState.onAdDisplayed() adState.onAdDisplayed(INSERT)
//计数 //计数
LimitUtils.addDisplayNum() LimitUtils.addDisplayNum()
...@@ -105,7 +114,7 @@ class AdOpenMgr { ...@@ -105,7 +114,7 @@ class AdOpenMgr {
showCallBack = null showCallBack = null
adState.onAdDisplayFailed() adState.onAdDisplayFailed()
adEvent?.adShowError(adError) admobEvent.adShowError(adError)
} }
...@@ -115,15 +124,14 @@ class AdOpenMgr { ...@@ -115,15 +124,14 @@ class AdOpenMgr {
showCallBack?.close() showCallBack?.close()
showCallBack = null showCallBack = null
adState.onAdHidden() adState.onAdHidden(INSERT)
//预加载,“Timeout for show call succeed.”预加载的广告大概率, //预加载,“Timeout for show call succeed.”预加载的广告大概率,
loadAd(MyApplication.appContext, AdmobEvent("openAd", "preload")) loadAd(MyApplication.appContext, AdmobEvent("openAd", "preload"))
} }
override fun onAdClicked() { override fun onAdClicked() {
adEvent?.clickAd(this@run.responseInfo) admobEvent.clickAd(this@run.responseInfo)
//计数 //计数
LimitUtils.addClickNum() LimitUtils.addClickNum()
} }
...@@ -135,7 +143,7 @@ class AdOpenMgr { ...@@ -135,7 +143,7 @@ class AdOpenMgr {
fun loadAd( fun loadAd(
context: Context, context: Context,
adEvent: AdEvent, adEvent: AdEvent,
loadCallBack: (() -> Unit)? = null loadCallBack: ((flag: Boolean) -> Unit)? = null
) { ) {
if (!adEvent.isUnLimit) { if (!adEvent.isUnLimit) {
...@@ -148,14 +156,12 @@ class AdOpenMgr { ...@@ -148,14 +156,12 @@ class AdOpenMgr {
} }
//避免无效预加载 //避免无效预加载
if (adState.loadingAd && loadCallBack == null && loadingCallBack == null) { if (adState.loadingAd) {
//容错机制
adState.loadingAd = false
return return
} }
adState.loadingAd = true
adEvent.adPulStart() adEvent.adPulStart()
adState.loadStart(adEvent)
AppOpenAd.load( AppOpenAd.load(
context, context,
...@@ -163,21 +169,27 @@ class AdOpenMgr { ...@@ -163,21 +169,27 @@ class AdOpenMgr {
AdRequest.Builder().build(), AdRequest.Builder().build(),
object : AppOpenAd.AppOpenAdLoadCallback() { object : AppOpenAd.AppOpenAdLoadCallback() {
override fun onAdLoaded(appOpenAd: AppOpenAd) { override fun onAdLoaded(appOpenAd: AppOpenAd) {
adState.onAdLoaded(appOpenAd, adEvent) LogEx.logDebug(adEvent.TAG, "open onAdLoaded loadAd")
loadCallBack?.invoke() appOpenAd.onPaidEventListener = AdmobOnPaidEventListener(appOpenAd, adEvent.scope)
loadingCallBack?.invoke() adState.onAdLoaded(appOpenAd)
loadCallBack?.invoke(true)
loadingCallBack?.invoke(true)
loadingCallBack = null loadingCallBack = null
(adEvent as AdmobEvent).pullAd(appOpenAd.responseInfo)
appOpenAd.onPaidEventListener = AdmobOnPaidEventListener(appOpenAd, adEvent.scope)
LimitUtils.addRequestNum() LimitUtils.addRequestNum()
(adEvent as AdmobEvent).pullAd(appOpenAd.responseInfo)
} }
override fun onAdFailedToLoad(loadAdError: LoadAdError) { override fun onAdFailedToLoad(loadAdError: LoadAdError) {
showCallBack?.adFailed() LogEx.logDebug(adEvent.TAG, "open onAdFailedToLoad loadAd")
showCallBack = null
adState.onAdLoadFailed() adState.onAdLoadFailed()
loadCallBack?.invoke(false)
loadingCallBack?.invoke(false)
loadingCallBack = null
(adEvent as AdmobEvent).pullAd(loadAdError.responseInfo, loadAdError) (adEvent as AdmobEvent).pullAd(loadAdError.responseInfo, loadAdError)
} }
} }
......
...@@ -49,6 +49,8 @@ class AdmobEvent : AdEvent { ...@@ -49,6 +49,8 @@ class AdmobEvent : AdEvent {
) { ) {
var key = "ad_pull" var key = "ad_pull"
val obj = JSONObject() val obj = JSONObject()
obj.put("ad_unit", adUnit)
obj.put("from", from)
obj.put("req_id", reqId) obj.put("req_id", reqId)
if (responseInfo != null) { if (responseInfo != null) {
val response = responseInfo.adapterResponses.getOrNull(0) val response = responseInfo.adapterResponses.getOrNull(0)
...@@ -65,8 +67,6 @@ class AdmobEvent : AdEvent { ...@@ -65,8 +67,6 @@ class AdmobEvent : AdEvent {
obj.put("session_id", responseInfo.responseId) obj.put("session_id", responseInfo.responseId)
} }
obj.put("networkname", responseInfo?.mediationAdapterClassName) obj.put("networkname", responseInfo?.mediationAdapterClassName)
obj.put("ad_unit", adUnit)
obj.put("from", from)
if (error == null) { if (error == null) {
obj.put("status", "1") obj.put("status", "1")
} else { } else {
...@@ -75,7 +75,7 @@ class AdmobEvent : AdEvent { ...@@ -75,7 +75,7 @@ class AdmobEvent : AdEvent {
key = "ad_pull_error" key = "ad_pull_error"
} }
EventUtils.event(key, ext = obj) EventUtils.event(key, ext = obj)
LogEx.logDebug(TAG, "ad_pull obj=$obj") LogEx.logDebug(TAG, "${key}_$adUnit obj=$obj")
} }
......
package com.dumpster.cleaner.business.ads.applovin //package com.dumpster.cleaner.business.ads.applovin
//
import android.os.Bundle //import android.os.Bundle
import com.applovin.mediation.MaxAd //import com.applovin.mediation.MaxAd
import com.applovin.mediation.MaxAdRevenueListener //import com.applovin.mediation.MaxAdRevenueListener
import com.applovin.mediation.MaxError //import com.applovin.mediation.MaxError
import com.applovin.sdk.AppLovinSdk //import com.applovin.sdk.AppLovinSdk
import com.dumpster.cleaner.MyApplication //import com.dumpster.cleaner.MyApplication
import com.dumpster.cleaner.business.ads.AdEvent //import com.dumpster.cleaner.business.ads.AdEvent
import com.dumpster.cleaner.business.ads.taichiPref //import com.dumpster.cleaner.business.ads.taichiPref
import com.dumpster.cleaner.business.ads.taichiSharedPreferencesEditor //import com.dumpster.cleaner.business.ads.taichiSharedPreferencesEditor
import com.dumpster.cleaner.business.helper.EventUtils //import com.dumpster.cleaner.business.helper.EventUtils
import com.dumpster.cleaner.utils.LogEx.logDebug //import com.dumpster.cleaner.utils.LogEx.logDebug
import com.facebook.appevents.AppEventsConstants //import com.facebook.appevents.AppEventsConstants
import com.facebook.appevents.AppEventsLogger //import com.facebook.appevents.AppEventsLogger
import com.google.firebase.analytics.FirebaseAnalytics //import com.google.firebase.analytics.FirebaseAnalytics
import org.json.JSONObject //import org.json.JSONObject
//
//
class AdMaxEvent : AdEvent { //class AdMaxEvent : AdEvent {
//
override val TAG: String = "AdMaxEvent" // override val TAG: String = "AdMaxEvent"
//
constructor(adUnit: String, from: String) : super() { // constructor(adUnit: String, from: String) : super() {
this.adUnit = adUnit // this.adUnit = adUnit
this.from = from // this.from = from
} // }
//
fun pullAd( // fun pullAd(
ad: MaxAd?, // ad: MaxAd?,
error: MaxError? = null // error: MaxError? = null
) { // ) {
val obj = JSONObject() // val obj = JSONObject()
obj.put("UnitId", ad?.adUnitId) // obj.put("UnitId", ad?.adUnitId)
obj.put("ad_unit", adUnit) // obj.put("ad_unit", adUnit)
obj.put( // obj.put(
"creativeId", // "creativeId",
ad?.creativeId // ad?.creativeId
) // )
obj.put("req_id", reqId) // obj.put("req_id", reqId)
obj.put("from", from) // obj.put("from", from)
obj.put("status", if (ad == null) "0" else "1") // obj.put("status", if (ad == null) "0" else "1")
obj.put("networkname", ad?.networkName) // obj.put("networkname", ad?.networkName)
obj.put("placement", ad?.placement) // obj.put("placement", ad?.placement)
obj.put("networkplacement", ad?.networkPlacement) // obj.put("networkplacement", ad?.networkPlacement)
obj.put("latency", ad?.requestLatencyMillis) // obj.put("latency", ad?.requestLatencyMillis)
obj.put("valueMicros", ad?.revenue) // obj.put("valueMicros", ad?.revenue)
if (error == null) { // if (error == null) {
obj.put("status", "1") // obj.put("status", "1")
} else { // } else {
obj.put("errMsg", error) // obj.put("errMsg", error)
obj.put("status", "2") // obj.put("status", "2")
} // }
EventUtils.event("ad_pull", ext = obj) // EventUtils.event("ad_pull", ext = obj)
logDebug(TAG, "ad_pull $obj") // logDebug(TAG, "ad_pull $obj")
} // }
//
fun clickAd(ad: MaxAd?) { // fun clickAd(ad: MaxAd?) {
//
val obj = JSONObject() // val obj = JSONObject()
obj.put("UnitId", ad?.adUnitId) // obj.put("UnitId", ad?.adUnitId)
obj.put("ad_unit", adUnit) // obj.put("ad_unit", adUnit)
obj.put( // obj.put(
"creativeId", // "creativeId",
ad?.creativeId // ad?.creativeId
) // )
obj.put("networkname", ad?.networkName) // obj.put("networkname", ad?.networkName)
obj.put("placement", ad?.placement) // obj.put("placement", ad?.placement)
obj.put("networkplacement", ad?.networkPlacement) // obj.put("networkplacement", ad?.networkPlacement)
obj.put("latency", ad?.requestLatencyMillis) // obj.put("latency", ad?.requestLatencyMillis)
obj.put("valueMicros", ad?.revenue) // obj.put("valueMicros", ad?.revenue)
if (!adUnit.equals("nativeAd")) { // if (!adUnit.equals("nativeAd")) {
EventUtils.event("ad_click", ext = obj) // EventUtils.event("ad_click", ext = obj)
} else { // } else {
EventUtils.event("ad_click", ext = obj) // EventUtils.event("ad_click", ext = obj)
} // }
//
} // }
//
fun showAd(ad: MaxAd?, activity: String?) { // fun showAd(ad: MaxAd?, activity: String?) {
val obj = JSONObject() // val obj = JSONObject()
obj.put("UnitId", ad?.adUnitId) // obj.put("UnitId", ad?.adUnitId)
obj.put("ad_unit", adUnit) // obj.put("ad_unit", adUnit)
obj.put( // obj.put(
"creativeId", // "creativeId",
ad?.creativeId // ad?.creativeId
) // )
obj.put("networkname", ad?.networkName) // obj.put("networkname", ad?.networkName)
obj.put("placement", ad?.placement) // obj.put("placement", ad?.placement)
obj.put("networkplacement", ad?.networkPlacement) // obj.put("networkplacement", ad?.networkPlacement)
obj.put("latency", ad?.requestLatencyMillis) // obj.put("latency", ad?.requestLatencyMillis)
obj.put("valueMicros", ad?.revenue) // obj.put("valueMicros", ad?.revenue)
obj.put("from", activity) // obj.put("from", activity)
obj.put("mediation", "applovin") // obj.put("mediation", "applovin")
if (adUnit != "nativeAd") { // if (adUnit != "nativeAd") {
EventUtils.event("ad_show", ext = obj) // EventUtils.event("ad_show", ext = obj)
} else { // } else {
EventUtils.event("ad_show", ext = obj) // EventUtils.event("ad_show", ext = obj)
} // }
//
} // }
//
//
class EventOnPaidEventListener : MaxAdRevenueListener { // class EventOnPaidEventListener : MaxAdRevenueListener {
override fun onAdRevenuePaid(ad: MaxAd) { // override fun onAdRevenuePaid(ad: MaxAd) {
val params = Bundle() // val params = Bundle()
val currentImpressionRevenue: Double = ad.revenue // In USD // val currentImpressionRevenue: Double = ad.revenue // In USD
val mFirebaseAnalytics = FirebaseAnalytics.getInstance(MyApplication.appContext) // val mFirebaseAnalytics = FirebaseAnalytics.getInstance(MyApplication.appContext)
params.putString(FirebaseAnalytics.Param.AD_PLATFORM, "appLovin") // params.putString(FirebaseAnalytics.Param.AD_PLATFORM, "appLovin")
params.putString(FirebaseAnalytics.Param.AD_SOURCE, ad.networkName) // params.putString(FirebaseAnalytics.Param.AD_SOURCE, ad.networkName)
params.putString(FirebaseAnalytics.Param.AD_FORMAT, ad.format.getDisplayName()) // params.putString(FirebaseAnalytics.Param.AD_FORMAT, ad.format.getDisplayName())
params.putString(FirebaseAnalytics.Param.AD_UNIT_NAME, ad.adUnitId) // params.putString(FirebaseAnalytics.Param.AD_UNIT_NAME, ad.adUnitId)
params.putDouble(FirebaseAnalytics.Param.VALUE, currentImpressionRevenue) // params.putDouble(FirebaseAnalytics.Param.VALUE, currentImpressionRevenue)
params.putString(FirebaseAnalytics.Param.CURRENCY, "USD") // params.putString(FirebaseAnalytics.Param.CURRENCY, "USD")
mFirebaseAnalytics.logEvent(FirebaseAnalytics.Event.AD_IMPRESSION, params) // mFirebaseAnalytics.logEvent(FirebaseAnalytics.Event.AD_IMPRESSION, params)
mFirebaseAnalytics.logEvent("Ad_Impression_Revenue", params) // mFirebaseAnalytics.logEvent("Ad_Impression_Revenue", params)
val previousTaichiTroasCache = taichiPref.getFloat("TaichiTroasCache", 0f) // val previousTaichiTroasCache = taichiPref.getFloat("TaichiTroasCache", 0f)
val currentTaichiTroasCache = previousTaichiTroasCache + currentImpressionRevenue // val currentTaichiTroasCache = previousTaichiTroasCache + currentImpressionRevenue
if (currentTaichiTroasCache >= 0.01) { // if (currentTaichiTroasCache >= 0.01) {
val roasbundle = Bundle() // val roasbundle = Bundle()
roasbundle.putDouble(FirebaseAnalytics.Param.VALUE, currentTaichiTroasCache) // roasbundle.putDouble(FirebaseAnalytics.Param.VALUE, currentTaichiTroasCache)
roasbundle.putString(FirebaseAnalytics.Param.CURRENCY, "USD")///(Required)tROAS事件必须 // roasbundle.putString(FirebaseAnalytics.Param.CURRENCY, "USD")///(Required)tROAS事件必须
mFirebaseAnalytics.logEvent("Total_Ads_Revenue_001", roasbundle) // 给Taichi用 // mFirebaseAnalytics.logEvent("Total_Ads_Revenue_001", roasbundle) // 给Taichi用
taichiSharedPreferencesEditor.putFloat("TaichiTroasCache", 0f)//重新清零,开始计算 // taichiSharedPreferencesEditor.putFloat("TaichiTroasCache", 0f)//重新清零,开始计算
//
val logger = AppEventsLogger.newLogger(MyApplication.appContext) // val logger = AppEventsLogger.newLogger(MyApplication.appContext)
val parameters = Bundle() // val parameters = Bundle()
parameters.putString(AppEventsConstants.EVENT_PARAM_CURRENCY, "USD") // parameters.putString(AppEventsConstants.EVENT_PARAM_CURRENCY, "USD")
logger.logEvent("ad_value", currentTaichiTroasCache, parameters) // logger.logEvent("ad_value", currentTaichiTroasCache, parameters)
} else { // } else {
taichiSharedPreferencesEditor.putFloat( // taichiSharedPreferencesEditor.putFloat(
"TaichiTroasCache", // "TaichiTroasCache",
currentTaichiTroasCache.toFloat() // currentTaichiTroasCache.toFloat()
) // )
taichiSharedPreferencesEditor.commit() // taichiSharedPreferencesEditor.commit()
} // }
val obj = JSONObject() // val obj = JSONObject()
val revenue = ad.revenue // val revenue = ad.revenue
val countryCode = // val countryCode =
AppLovinSdk.getInstance(MyApplication.appContext).configuration.countryCode // AppLovinSdk.getInstance(MyApplication.appContext).configuration.countryCode
val networkName = ad.networkName // val networkName = ad.networkName
val adUnitId = ad.adUnitId // val adUnitId = ad.adUnitId
val adFormat = ad.format // val adFormat = ad.format
val placement = ad.placement // val placement = ad.placement
val networkPlacement = ad.networkPlacement // val networkPlacement = ad.networkPlacement
obj.put("valueMicros", revenue) // obj.put("valueMicros", revenue)
obj.put("currencyCode", countryCode) // obj.put("currencyCode", countryCode)
obj.put("adUnitId", adUnitId) // obj.put("adUnitId", adUnitId)
obj.put("networkName", networkName) // obj.put("networkName", networkName)
obj.put("adFormat", adFormat) // obj.put("adFormat", adFormat)
obj.put("placement", placement) // obj.put("placement", placement)
obj.put("networkPlacement", networkPlacement) // obj.put("networkPlacement", networkPlacement)
EventUtils.event("ad_price", ext = obj) // EventUtils.event("ad_price", ext = obj)
} // }
} // }
//
} //}
\ No newline at end of file \ No newline at end of file
package com.dumpster.cleaner.business.ads.applovin //package com.dumpster.cleaner.business.ads.applovin
//
import android.app.Activity //import android.app.Activity
import com.applovin.mediation.MaxAd //import com.applovin.mediation.MaxAd
import com.applovin.mediation.MaxAdListener //import com.applovin.mediation.MaxAdListener
import com.applovin.mediation.MaxError //import com.applovin.mediation.MaxError
import com.applovin.mediation.ads.MaxInterstitialAd //import com.applovin.mediation.ads.MaxInterstitialAd
import com.dumpster.cleaner.GlobalConfig //import com.dumpster.cleaner.GlobalConfig
import com.dumpster.cleaner.business.ads.AdCountDownDialog.showAdCountDownDialog //import com.dumpster.cleaner.business.ads.AdCountDownDialog.showAdCountDownDialog
import com.dumpster.cleaner.business.ads.AdEvent //import com.dumpster.cleaner.business.ads.AdEvent
import com.dumpster.cleaner.business.ads.AdState //import com.dumpster.cleaner.business.ads.AdState
import com.dumpster.cleaner.business.ads.AdsShowCallBack //import com.dumpster.cleaner.business.ads.AdsShowCallBack
import com.dumpster.cleaner.business.ads.AdsType //import com.dumpster.cleaner.business.ads.AdsType
import com.dumpster.cleaner.business.ads.LimitUtils //import com.dumpster.cleaner.business.ads.LimitUtils
//
/** ///**
*插屏广告加载显示管理类 // *插屏广告加载显示管理类
*/ // */
class MaxInsertMgr { //class MaxInsertMgr {
//
private var adState = AdState<MaxInterstitialAd>() // private var adState = AdState<MaxInterstitialAd>()
private var showCallBack: AdsShowCallBack? = null // private var showCallBack: AdsShowCallBack? = null
//
fun show( // fun show(
activity: Activity, // activity: Activity,
isUnLimit: Boolean, // isUnLimit: Boolean,
adEvent: AdEvent, // adEvent: AdEvent,
showCallBack: AdsShowCallBack? // showCallBack: AdsShowCallBack?
) { // ) {
//
if (activity.isFinishing || activity.isDestroyed) { // if (activity.isFinishing || activity.isDestroyed) {
showCallBack?.failed(1) // showCallBack?.failed(1)
return // return
} // }
//
if (showCallBack != null) { // if (showCallBack != null) {
this.showCallBack = showCallBack // this.showCallBack = showCallBack
if (adState.adDialog == null) { // if (adState.adDialog == null) {
adState.adDialog = activity.showAdCountDownDialog() // adState.adDialog = activity.showAdCountDownDialog()
} // }
adEvent.adPrepareShow() // adEvent.adPrepareShow()
} // }
//
if (!adState.loadingAd) { // if (!adState.loadingAd) {
if (!isUnLimit) { // if (!isUnLimit) {
if (!LimitUtils.isAdShow(AdsType.INSERT, adEvent)) { // if (!LimitUtils.isAdShow(AdsType.INSERT, adEvent)) {
showCallBack?.failed(3) // showCallBack?.failed(3)
return // return
} // }
if (LimitUtils.isIntervalLimited(adEvent)) { // if (LimitUtils.isIntervalLimited(adEvent)) {
showCallBack?.failed(4) // showCallBack?.failed(4)
return // return
} // }
} // }
//
if (!adAvailable() || adState.currentAd == null) { // if (!adAvailable() || adState.currentAd == null) {
loadAd(activity, adEvent, isUnLimit) // loadAd(activity, adEvent, isUnLimit)
return // return
} // }
//
if (adState.currentAd?.isReady == false) { // if (adState.currentAd?.isReady == false) {
loadAd(activity, adEvent, isUnLimit) // loadAd(activity, adEvent, isUnLimit)
return // return
} // }
showReadyAd(adEvent) // showReadyAd(adEvent)
} // }
//
} // }
//
//
private fun showReadyAd(adEvent: AdEvent) { // private fun showReadyAd(adEvent: AdEvent) {
adState.currentAd?.run { // adState.currentAd?.run {
setListener(object : MaxAdListener { // setListener(object : MaxAdListener {
override fun onAdLoaded(p0: MaxAd) = Unit // override fun onAdLoaded(p0: MaxAd) = Unit
override fun onAdLoadFailed(p0: String, p1: MaxError) = Unit // override fun onAdLoadFailed(p0: String, p1: MaxError) = Unit
//
override fun onAdDisplayed(ad: MaxAd) { // override fun onAdDisplayed(ad: MaxAd) {
//
adState.onAdDisplayed() // adState.onAdDisplayed()
showCallBack?.show() // showCallBack?.show()
//
(adEvent as AdMaxEvent).showAd(ad, activity::class.simpleName) // (adEvent as AdMaxEvent).showAd(ad, activity::class.simpleName)
//计数 // //计数
LimitUtils.addDisplayNum() // LimitUtils.addDisplayNum()
} // }
//
override fun onAdDisplayFailed(ad: MaxAd, error: MaxError) { // override fun onAdDisplayFailed(ad: MaxAd, error: MaxError) {
adState.onAdDisplayFailed() // adState.onAdDisplayFailed()
showCallBack?.adFailed() // showCallBack?.adFailed()
showCallBack = null // showCallBack = null
//
(adEvent as AdMaxEvent).adShowError(error) // (adEvent as AdMaxEvent).adShowError(error)
} // }
//
override fun onAdHidden(p0: MaxAd) { // override fun onAdHidden(p0: MaxAd) {
//
adState.onAdHidden() // adState.onAdHidden()
showCallBack?.close() // showCallBack?.close()
//
loadAd(activity, AdMaxEvent("interAd", "preload")) // loadAd(activity, AdMaxEvent("interAd", "preload"))
} // }
//
override fun onAdClicked(ad: MaxAd) { // override fun onAdClicked(ad: MaxAd) {
(adEvent as AdMaxEvent).clickAd(ad) // (adEvent as AdMaxEvent).clickAd(ad)
//计数 // //计数
LimitUtils.addClickNum() // LimitUtils.addClickNum()
} // }
//
//
}) // })
setRevenueListener(AdMaxEvent.EventOnPaidEventListener()) // setRevenueListener(AdMaxEvent.EventOnPaidEventListener())
//
showAd(activity) // showAd(activity)
} // }
} // }
//
//
fun loadAd( // fun loadAd(
ac: Activity, // ac: Activity,
adEvent: AdEvent, // adEvent: AdEvent,
isUnLimit: Boolean = false, // isUnLimit: Boolean = false,
) { // ) {
if (!isUnLimit) { // if (!isUnLimit) {
if (!LimitUtils.isAdShow(AdsType.INSERT, adEvent)) { // if (!LimitUtils.isAdShow(AdsType.INSERT, adEvent)) {
this.showCallBack?.close(4) // this.showCallBack?.close(4)
this.showCallBack = null // this.showCallBack = null
return // return
} // }
} // }
//
if (!adState.loadingAd) { // if (!adState.loadingAd) {
adState.loadingAd = true // adState.loadingAd = true
//
adEvent.adPulStart() // adEvent.adPulStart()
//
adState.currentAd = MaxInterstitialAd(GlobalConfig.ID_MAX_INTER, ac) // adState.currentAd = MaxInterstitialAd(GlobalConfig.ID_MAX_INTER, ac)
adState.currentAd?.setListener(object : MaxAdListener { // adState.currentAd?.setListener(object : MaxAdListener {
//
override fun onAdDisplayed(p0: MaxAd) = Unit // override fun onAdDisplayed(p0: MaxAd) = Unit
override fun onAdHidden(p0: MaxAd) = Unit // override fun onAdHidden(p0: MaxAd) = Unit
override fun onAdClicked(p0: MaxAd) = Unit // override fun onAdClicked(p0: MaxAd) = Unit
override fun onAdDisplayFailed(p0: MaxAd, p1: MaxError) = Unit // override fun onAdDisplayFailed(p0: MaxAd, p1: MaxError) = Unit
//
override fun onAdLoaded(ad: MaxAd) { // override fun onAdLoaded(ad: MaxAd) {
adState.onAdLoaded(null, adEvent) // adState.onAdLoaded(null, adEvent)
//
show(ac, isUnLimit, adEvent, null) // show(ac, isUnLimit, adEvent, null)
//
(adEvent as AdMaxEvent).pullAd(ad) // (adEvent as AdMaxEvent).pullAd(ad)
LimitUtils.addRequestNum() // LimitUtils.addRequestNum()
} // }
//
override fun onAdLoadFailed(ad: String, error: MaxError) { // override fun onAdLoadFailed(ad: String, error: MaxError) {
adState.onAdLoadFailed() // adState.onAdLoadFailed()
//
(adEvent as AdMaxEvent).pullAd(null, error) // (adEvent as AdMaxEvent).pullAd(null, error)
//
showCallBack?.adFailed(5) // showCallBack?.adFailed(5)
showCallBack = null // showCallBack = null
} // }
//
}) // })
adState.currentAd?.loadAd() // adState.currentAd?.loadAd()
} // }
} // }
//
//
private fun adAvailable() = ((System.currentTimeMillis() - adState.lastLoadTime) / 1000 / 60).toInt() < 30 // private fun adAvailable() = ((System.currentTimeMillis() - adState.lastLoadTime) / 1000 / 60).toInt() < 30
} //}
\ No newline at end of file \ No newline at end of file
package com.dumpster.cleaner.business.ads.applovin //package com.dumpster.cleaner.business.ads.applovin
//
import androidx.annotation.LayoutRes //import androidx.annotation.LayoutRes
import com.applovin.mediation.MaxAd //import com.applovin.mediation.MaxAd
import com.applovin.mediation.MaxError //import com.applovin.mediation.MaxError
import com.applovin.mediation.nativeAds.MaxNativeAdListener //import com.applovin.mediation.nativeAds.MaxNativeAdListener
import com.applovin.mediation.nativeAds.MaxNativeAdLoader //import com.applovin.mediation.nativeAds.MaxNativeAdLoader
import com.applovin.mediation.nativeAds.MaxNativeAdView //import com.applovin.mediation.nativeAds.MaxNativeAdView
import com.dumpster.cleaner.GlobalConfig //import com.dumpster.cleaner.GlobalConfig
import com.dumpster.cleaner.business.ads.AdsType //import com.dumpster.cleaner.business.ads.AdsType
import com.dumpster.cleaner.business.ads.LimitUtils //import com.dumpster.cleaner.business.ads.LimitUtils
import com.dumpster.cleaner.business.ads.NativeParentView //import com.dumpster.cleaner.business.ads.NativeParentView
import com.dumpster.cleaner.business.helper.EventUtils //import com.dumpster.cleaner.business.helper.EventUtils
import org.json.JSONObject //import org.json.JSONObject
import java.util.UUID //import java.util.UUID
//
/** ///**
*原生广告加载显示管理类 // *原生广告加载显示管理类
*/ // */
class MaxNativeMgr { //class MaxNativeMgr {
//
/** // /**
* 上一次的缓存成功时间 // * 上一次的缓存成功时间
*/ // */
protected var lastTime: Long = 0 // protected var lastTime: Long = 0
//
/** // /**
* 原生广告 // * 原生广告
*/ // */
private var currentAd: MaxAd? = null // private var currentAd: MaxAd? = null
private var currentLoader: MaxNativeAdLoader? = null // private var currentLoader: MaxNativeAdLoader? = null
//
//
private fun loadAd( // private fun loadAd(
adMaxEvent: AdMaxEvent, // adMaxEvent: AdMaxEvent,
parent: NativeParentView, // parent: NativeParentView,
@LayoutRes layout: Int // @LayoutRes layout: Int
) { // ) {
//
if (!LimitUtils.isAdShow(AdsType.NATIVE, adMaxEvent)) return // if (!LimitUtils.isAdShow(AdsType.NATIVE, adMaxEvent)) return
//
val reqId = UUID.randomUUID().toString() // val reqId = UUID.randomUUID().toString()
val obj = JSONObject() // val obj = JSONObject()
obj.put("req_id", reqId) // obj.put("req_id", reqId)
obj.put("ad_type", "nativeAd") // obj.put("ad_type", "nativeAd")
//
val nativeAdLoader = MaxNativeAdLoader(GlobalConfig.ID_MAX_NATIVE, parent.context) // val nativeAdLoader = MaxNativeAdLoader(GlobalConfig.ID_MAX_NATIVE, parent.context)
//
nativeAdLoader.setNativeAdListener(object : MaxNativeAdListener() { // nativeAdLoader.setNativeAdListener(object : MaxNativeAdListener() {
//
override fun onNativeAdLoaded(nativeAdView: MaxNativeAdView?, ad: MaxAd) { // override fun onNativeAdLoaded(nativeAdView: MaxNativeAdView?, ad: MaxAd) {
currentLoader = nativeAdLoader // currentLoader = nativeAdLoader
currentAd = ad // currentAd = ad
lastTime = System.currentTimeMillis() // lastTime = System.currentTimeMillis()
adMaxEvent.pullAd(ad) // adMaxEvent.pullAd(ad)
nativeAdLoader.setRevenueListener(AdMaxEvent.EventOnPaidEventListener()) // nativeAdLoader.setRevenueListener(AdMaxEvent.EventOnPaidEventListener())
show(adMaxEvent, parent, layout) // show(adMaxEvent, parent, layout)
} // }
//
override fun onNativeAdLoadFailed(adUnitId: String, error: MaxError) { // override fun onNativeAdLoadFailed(adUnitId: String, error: MaxError) {
adMaxEvent.pullAd(null, error) // adMaxEvent.pullAd(null, error)
} // }
//
override fun onNativeAdClicked(ad: MaxAd) { // override fun onNativeAdClicked(ad: MaxAd) {
//
} // }
//
}) // })
nativeAdLoader.loadAd() // nativeAdLoader.loadAd()
} // }
//
fun show( // fun show(
adMaxEvent: AdMaxEvent, // adMaxEvent: AdMaxEvent,
parent: NativeParentView, // parent: NativeParentView,
@LayoutRes layout: Int, // @LayoutRes layout: Int,
nativeCallBack: ((Any?) -> Unit)? = null // nativeCallBack: ((Any?) -> Unit)? = null
) { // ) {
if (!LimitUtils.isAdShow(AdsType.NATIVE, adMaxEvent)) { // if (!LimitUtils.isAdShow(AdsType.NATIVE, adMaxEvent)) {
currentLoader = null // currentLoader = null
currentAd = null // currentAd = null
return // return
} // }
val nativeAd = currentAd // val nativeAd = currentAd
val nativeLoader = currentLoader // val nativeLoader = currentLoader
if ((nativeAd == null || nativeLoader == null).also { // if ((nativeAd == null || nativeLoader == null).also {
if (it) { // if (it) {
val obj2 = JSONObject() // val obj2 = JSONObject()
obj2.put("reason", "no_ad") // obj2.put("reason", "no_ad")
obj2.put("ad_unit", "nativeAd") // obj2.put("ad_unit", "nativeAd")
EventUtils.event("ad_show_error", ext = obj2) // EventUtils.event("ad_show_error", ext = obj2)
} // }
} || (!adAvailable()).also { // } || (!adAvailable()).also {
if (it) { // if (it) {
val obj2 = JSONObject() // val obj2 = JSONObject()
obj2.put("ad_unit", "nativeAd") // obj2.put("ad_unit", "nativeAd")
EventUtils.event("ad_expire", ext = obj2) // EventUtils.event("ad_expire", ext = obj2)
} // }
}) { // }) {
//缓存过期了就清空 // //缓存过期了就清空
currentLoader = null // currentLoader = null
currentAd = null // currentAd = null
loadAd(AdMaxEvent("nativeAd", "preload"), parent, layout) // loadAd(AdMaxEvent("nativeAd", "preload"), parent, layout)
return // return
} // }
val obj = JSONObject() // val obj = JSONObject()
obj.put("ad_unit", "nativeAd") // obj.put("ad_unit", "nativeAd")
EventUtils.event("ad_prepare_show", ext = obj) // EventUtils.event("ad_prepare_show", ext = obj)
parent.setNativeAd(nativeLoader!!, nativeAd!!, layout) // parent.setNativeAd(nativeLoader!!, nativeAd!!, layout)
nativeCallBack?.invoke(nativeAd) // nativeCallBack?.invoke(nativeAd)
} // }
//
private fun adAvailable(): Boolean { // private fun adAvailable(): Boolean {
return ((System.currentTimeMillis() - lastTime) / 1000 / 60).toInt() < 30 // return ((System.currentTimeMillis() - lastTime) / 1000 / 60).toInt() < 30
} // }
} //}
\ No newline at end of file \ No newline at end of file
package com.dumpster.cleaner.business.ads.applovin //package com.dumpster.cleaner.business.ads.applovin
//
import android.app.Activity //import android.app.Activity
import android.content.Context //import android.content.Context
import com.applovin.mediation.MaxAd //import com.applovin.mediation.MaxAd
import com.applovin.mediation.MaxAdListener //import com.applovin.mediation.MaxAdListener
import com.applovin.mediation.MaxError //import com.applovin.mediation.MaxError
import com.applovin.mediation.ads.MaxAppOpenAd //import com.applovin.mediation.ads.MaxAppOpenAd
import com.dumpster.cleaner.GlobalConfig //import com.dumpster.cleaner.GlobalConfig
import com.dumpster.cleaner.business.ads.AdEvent //import com.dumpster.cleaner.business.ads.AdEvent
import com.dumpster.cleaner.business.ads.AdState //import com.dumpster.cleaner.business.ads.AdState
import com.dumpster.cleaner.business.ads.AdsShowCallBack //import com.dumpster.cleaner.business.ads.AdsShowCallBack
import com.dumpster.cleaner.business.ads.AdsType //import com.dumpster.cleaner.business.ads.AdsType
import com.dumpster.cleaner.business.ads.LimitUtils //import com.dumpster.cleaner.business.ads.LimitUtils
//
/** ///**
* 开屏广告加载显示管理类 // * 开屏广告加载显示管理类
*/ // */
class MaxOpenMgr { //class MaxOpenMgr {
//
private val adState = AdState<MaxAppOpenAd>() // private val adState = AdState<MaxAppOpenAd>()
private var showCallBack: AdsShowCallBack? = null // private var showCallBack: AdsShowCallBack? = null
//
fun show(activity: Activity, isUnLimit: Boolean, adEvent: AdEvent, showCallBack: AdsShowCallBack?) { // fun show(activity: Activity, isUnLimit: Boolean, adEvent: AdEvent, showCallBack: AdsShowCallBack?) {
if (activity.isFinishing || activity.isDestroyed) { // if (activity.isFinishing || activity.isDestroyed) {
return // return
} // }
//
if (showCallBack != null) { // if (showCallBack != null) {
this.showCallBack = showCallBack // this.showCallBack = showCallBack
adEvent.adPrepareShow() // adEvent.adPrepareShow()
} // }
//
if (!adState.loadingAd) { // if (!adState.loadingAd) {
//
if (!isUnLimit) { // if (!isUnLimit) {
if (!LimitUtils.isAdShow(AdsType.OPEN, adEvent)) { // if (!LimitUtils.isAdShow(AdsType.OPEN, adEvent)) {
showCallBack?.failed() // showCallBack?.failed()
return // return
} // }
if (LimitUtils.isIntervalLimited(adEvent)) { // if (LimitUtils.isIntervalLimited(adEvent)) {
showCallBack?.failed() // showCallBack?.failed()
return // return
} // }
} // }
//
if (!adAvailable() || adState.currentAd == null) { // if (!adAvailable() || adState.currentAd == null) {
loadAd(activity, isUnLimit, adEvent) // loadAd(activity, isUnLimit, adEvent)
return // return
} // }
//
if (adState.currentAd?.isReady != true) { // if (adState.currentAd?.isReady != true) {
loadAd(activity, isUnLimit, adEvent) // loadAd(activity, isUnLimit, adEvent)
return // return
} // }
showReadyAd(activity, adEvent) // showReadyAd(activity, adEvent)
} // }
//
//
} // }
//
//
private fun showReadyAd(activity: Activity, adEvent: AdEvent) { // private fun showReadyAd(activity: Activity, adEvent: AdEvent) {
adState.currentAd?.run { // adState.currentAd?.run {
setListener(object : MaxAdListener { // setListener(object : MaxAdListener {
override fun onAdLoaded(p0: MaxAd) { // override fun onAdLoaded(p0: MaxAd) {
//
} // }
//
override fun onAdDisplayed(ad: MaxAd) { // override fun onAdDisplayed(ad: MaxAd) {
adState.onAdDisplayed() // adState.onAdDisplayed()
showCallBack?.show() // showCallBack?.show()
//
LimitUtils.addDisplayNum() // LimitUtils.addDisplayNum()
(adEvent as AdMaxEvent).showAd(ad, activity::class.simpleName) // (adEvent as AdMaxEvent).showAd(ad, activity::class.simpleName)
} // }
//
override fun onAdHidden(p0: MaxAd) { // override fun onAdHidden(p0: MaxAd) {
adState.onAdHidden() // adState.onAdHidden()
showCallBack?.close() // showCallBack?.close()
showCallBack = null // showCallBack = null
//
loadAd(activity.applicationContext, false, AdMaxEvent("openAd", "preload")) // loadAd(activity.applicationContext, false, AdMaxEvent("openAd", "preload"))
} // }
//
override fun onAdClicked(ad: MaxAd) { // override fun onAdClicked(ad: MaxAd) {
(adEvent as AdMaxEvent).clickAd(ad) // (adEvent as AdMaxEvent).clickAd(ad)
//计数 // //计数
LimitUtils.addClickNum() // LimitUtils.addClickNum()
} // }
//
override fun onAdLoadFailed(p0: String, p1: MaxError) { // override fun onAdLoadFailed(p0: String, p1: MaxError) {
//
} // }
//
override fun onAdDisplayFailed(p0: MaxAd, error: MaxError) { // override fun onAdDisplayFailed(p0: MaxAd, error: MaxError) {
adState.onAdDisplayFailed() // adState.onAdDisplayFailed()
showCallBack?.adFailed() // showCallBack?.adFailed()
showCallBack = null // showCallBack = null
adEvent.adShowError(error) // adEvent.adShowError(error)
} // }
//
}) // })
//
showAd() // showAd()
} // }
} // }
//
fun loadAd(context: Context, isUnLimit: Boolean, adEvent: AdEvent) { // fun loadAd(context: Context, isUnLimit: Boolean, adEvent: AdEvent) {
//
if (!isUnLimit) { // if (!isUnLimit) {
if (!LimitUtils.isAdShow(AdsType.OPEN, adEvent)) { // if (!LimitUtils.isAdShow(AdsType.OPEN, adEvent)) {
this.showCallBack?.close() // this.showCallBack?.close()
this.showCallBack = null // this.showCallBack = null
return // return
} // }
} // }
//
if (!adState.loadingAd) { // if (!adState.loadingAd) {
adState.loadingAd = true // adState.loadingAd = true
//
adEvent.adPulStart() // adEvent.adPulStart()
//
adState.currentAd = MaxAppOpenAd(GlobalConfig.ID_MAX_OPEN, context) // adState.currentAd = MaxAppOpenAd(GlobalConfig.ID_MAX_OPEN, context)
adState.currentAd?.setListener(object : MaxAdListener { // adState.currentAd?.setListener(object : MaxAdListener {
override fun onAdLoaded(ad: MaxAd) { // override fun onAdLoaded(ad: MaxAd) {
// adState.onAdLoaded(null) //// adState.onAdLoaded(null)
// val ac = adState.activityRef?.get() //// val ac = adState.activityRef?.get()
// if (ac != null) { //// if (ac != null) {
// show(ac, isUnLimit, adEvent, null) //// show(ac, isUnLimit, adEvent, null)
// } //// }
(adEvent as AdMaxEvent).pullAd(ad) // (adEvent as AdMaxEvent).pullAd(ad)
LimitUtils.addRequestNum() // LimitUtils.addRequestNum()
} // }
//
override fun onAdDisplayed(p0: MaxAd) { // override fun onAdDisplayed(p0: MaxAd) {
//
} // }
//
override fun onAdHidden(p0: MaxAd) { // override fun onAdHidden(p0: MaxAd) {
//
} // }
//
override fun onAdClicked(p0: MaxAd) { // override fun onAdClicked(p0: MaxAd) {
} // }
//
override fun onAdLoadFailed(p0: String, error: MaxError) { // override fun onAdLoadFailed(p0: String, error: MaxError) {
adState.onAdLoadFailed() // adState.onAdLoadFailed()
showCallBack?.adFailed() // showCallBack?.adFailed()
showCallBack = null // showCallBack = null
(adEvent as AdMaxEvent).pullAd(null, error) // (adEvent as AdMaxEvent).pullAd(null, error)
} // }
//
override fun onAdDisplayFailed(p0: MaxAd, p1: MaxError) { // override fun onAdDisplayFailed(p0: MaxAd, p1: MaxError) {
//
} // }
//
}) // })
adState.currentAd?.loadAd() // adState.currentAd?.loadAd()
} // }
} // }
//
//
fun adAvailable() = ((System.currentTimeMillis() - adState.lastLoadTime) / 1000 / 60).toInt() < 30 // fun adAvailable() = ((System.currentTimeMillis() - adState.lastLoadTime) / 1000 / 60).toInt() < 30
} //}
\ No newline at end of file \ No newline at end of file
...@@ -17,6 +17,7 @@ import com.dumpster.cleaner.base.cleanFileBeans ...@@ -17,6 +17,7 @@ import com.dumpster.cleaner.base.cleanFileBeans
import com.dumpster.cleaner.bean.FeatureBean.Companion.JUNK_CLEAN import com.dumpster.cleaner.bean.FeatureBean.Companion.JUNK_CLEAN
import com.dumpster.cleaner.bean.FileBean import com.dumpster.cleaner.bean.FileBean
import com.dumpster.cleaner.business.ads.AdsMgr import com.dumpster.cleaner.business.ads.AdsMgr
import com.dumpster.cleaner.business.ads.AdsShowCallBack
import com.dumpster.cleaner.databinding.ActivityJunkCleanBinding import com.dumpster.cleaner.databinding.ActivityJunkCleanBinding
import com.dumpster.cleaner.databinding.ItemJunkCleanChildBinding import com.dumpster.cleaner.databinding.ItemJunkCleanChildBinding
import com.dumpster.cleaner.databinding.ItemJunkCleanGroupBinding import com.dumpster.cleaner.databinding.ItemJunkCleanGroupBinding
...@@ -133,6 +134,11 @@ class JunkCleanActivity : BaseActivity<ActivityJunkCleanBinding>(ActivityJunkCle ...@@ -133,6 +134,11 @@ class JunkCleanActivity : BaseActivity<ActivityJunkCleanBinding>(ActivityJunkCle
selectedFiles = list.flatMap { it.items.takeIf { it.isNotEmpty() }?.filter { it.isSelected } ?: emptyList() } selectedFiles = list.flatMap { it.items.takeIf { it.isNotEmpty() }?.filter { it.isSelected } ?: emptyList() }
total = selectedFiles.sumOf { it.length } total = selectedFiles.sumOf { it.length }
setClean() setClean()
AdsMgr.showInsert(this, true, showCallBack = object : AdsShowCallBack() {
override fun next() {
binding.idShouzhi.visibility = View.VISIBLE
}
})
} }
......
package com.dumpster.cleaner.ui.guide
import android.animation.Animator
import androidx.core.view.ViewCompat
import androidx.core.view.WindowInsetsCompat
import androidx.core.view.isVisible
import androidx.core.view.updatePadding
import com.dumpster.cleaner.R
import com.dumpster.cleaner.base.BaseActivity
import com.dumpster.cleaner.base.jumpAction
import com.dumpster.cleaner.bean.FeatureBean.Companion.JUNK_CLEAN
import com.dumpster.cleaner.business.ads.AdsMgr
import com.dumpster.cleaner.business.ads.AdsShowCallBack
import com.dumpster.cleaner.databinding.ActivityLayoutCleanGuideBinding
import com.dumpster.cleaner.ui.dialog.StoragePermissionDialog
import com.dumpster.cleaner.ui.dialog.permissionStorageJump
import com.dumpster.cleaner.ui.main.MainActivity
import com.dumpster.cleaner.utils.BarUtils
import com.dumpster.cleaner.utils.PermissionUtils.checkStorePermission
import com.dumpster.cleaner.utils.PermissionUtils.requestStoragePermission
class CleanGuideActivity : BaseActivity<ActivityLayoutCleanGuideBinding>(ActivityLayoutCleanGuideBinding::inflate) {
override fun useDefaultImmersive() {
ViewCompat.setOnApplyWindowInsetsListener(binding.root) { v, insets ->
val systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars())
v.setPadding(systemBars.left, 0, systemBars.right, systemBars.bottom)
insets
}
}
override fun initView() {
super.initView()
binding.root.updatePadding(top = BarUtils.getStatusBarHeight())
play1()
}
override fun initListener() {
binding.idSkip.setOnClickListener {
AdsMgr.showInsert(this, true, object : AdsShowCallBack() {
override fun next() {
goToAc(MainActivity::class.java)
finish()
}
})
}
binding.idClean02.setOnClickListener {
val jump = {
jumpAction(JUNK_CLEAN)
finish()
}
if (this.checkStorePermission()) {
jump.invoke()
} else {
val dialog = StoragePermissionDialog(this)
dialog.action = {
requestStoragePermission(launcher) {
if (it) {
jump.invoke()
}
}
}
dialog.showDialog()
}
}
}
private fun play1() {
binding.idLottieview.setAnimation(R.raw.firstscan)
binding.idLottieview.playAnimation()
binding.idLottieview.addAnimatorListener(createAnimatorListener())
binding.idLottieview.addAnimatorUpdateListener { animation ->
// 获取动画的当前进度(0.0f 到 1.0f)
val progress = animation.animatedFraction
// 将进度条的进度设置为动画进度的百分比
binding.idProg01.progress = (progress * 100).toInt()
if (binding.idProg01.progress == 50) {
play2()
}
}
binding.idProg01.progress = 0
binding.idProg01.max = 50
}
private fun play2() {
binding.idLottieview.setAnimation(R.raw.firstscan_finish)
binding.idLottieview.playAnimation()
binding.idLl01.isVisible = false
binding.idLl02.isVisible = true
binding.idSkip.isVisible = true
}
private fun createAnimatorListener(): Animator.AnimatorListener {
return object : Animator.AnimatorListener {
override fun onAnimationStart(p0: Animator) {
// 动画开始时调用
}
override fun onAnimationEnd(p0: Animator) {
// 动画结束时调用
}
override fun onAnimationCancel(p0: Animator) {
// 动画被取消时调用
}
override fun onAnimationRepeat(p0: Animator) {
// 动画重复时调用
}
}
}
override fun handleBackCallBack() {
binding.idSkip.callOnClick()
}
}
\ No newline at end of file
...@@ -35,7 +35,7 @@ import com.dumpster.cleaner.business.helper.NewComUtils.spConfig ...@@ -35,7 +35,7 @@ import com.dumpster.cleaner.business.helper.NewComUtils.spConfig
import com.dumpster.cleaner.business.service.StayJobService.Companion.startStayJobService import com.dumpster.cleaner.business.service.StayJobService.Companion.startStayJobService
import com.dumpster.cleaner.databinding.ActivitySplashBinding import com.dumpster.cleaner.databinding.ActivitySplashBinding
import com.dumpster.cleaner.ui.dialog.StoragePermissionDialog import com.dumpster.cleaner.ui.dialog.StoragePermissionDialog
import com.dumpster.cleaner.ui.guide.GuideCleanActivity import com.dumpster.cleaner.ui.guide.CleanGuideActivity
import com.dumpster.cleaner.ui.main.MainActivity import com.dumpster.cleaner.ui.main.MainActivity
import com.dumpster.cleaner.utils.LogEx import com.dumpster.cleaner.utils.LogEx
import com.dumpster.cleaner.utils.PermissionUtils.checkStorePermission import com.dumpster.cleaner.utils.PermissionUtils.checkStorePermission
...@@ -84,6 +84,9 @@ class SplashActivity : BaseActivity<ActivitySplashBinding>(ActivitySplashBinding ...@@ -84,6 +84,9 @@ class SplashActivity : BaseActivity<ActivitySplashBinding>(ActivitySplashBinding
requestNotificationPermission() requestNotificationPermission()
showLoadingAd() showLoadingAd()
setPrivacyPolicy() setPrivacyPolicy()
viewModel.jumpNext={
actionIdJump()
}
} }
private fun setPrivacyPolicy() { private fun setPrivacyPolicy() {
...@@ -235,7 +238,7 @@ class SplashActivity : BaseActivity<ActivitySplashBinding>(ActivitySplashBinding ...@@ -235,7 +238,7 @@ class SplashActivity : BaseActivity<ActivitySplashBinding>(ActivitySplashBinding
when { when {
!isGuided -> { !isGuided -> {
isGuided = true isGuided = true
goToAc(GuideCleanActivity::class.java) goToAc(CleanGuideActivity::class.java)
finish() finish()
} }
......
<?xml version="1.0" encoding="utf-8"?>
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
<item android:id="@android:id/background">
<shape>
<corners android:radius="5dp" />
<solid android:color="@color/white" />
</shape>
</item>
<item android:id="@android:id/progress">
<clip>
<shape>
<corners android:radius="5dp" />
<solid android:color="#6AFAE7" />
</shape>
</clip>
</item>
</layer-list>
\ No newline at end of file
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/colorPrimary">
<com.noober.background.view.BLTextView
android:id="@+id/id_skip"
android:layout_width="92dp"
android:layout_height="36dp"
android:layout_marginTop="36dp"
android:layout_marginEnd="20dp"
android:gravity="center"
android:text="Skip"
android:textColor="@color/colorPrimary"
android:textSize="16sp"
android:visibility="gone"
app:bl_corners_radius="50dp"
app:bl_solid_color="#B0FFFFFF"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<com.airbnb.lottie.LottieAnimationView
android:id="@+id/id_lottieview"
android:layout_width="match_parent"
android:layout_height="0dp"
app:layout_constraintBottom_toTopOf="@+id/id_fl_bottom"
app:layout_constraintTop_toTopOf="parent" />
<FrameLayout
android:id="@+id/id_fl_bottom"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="100dp"
app:layout_constraintBottom_toBottomOf="parent">
<androidx.appcompat.widget.LinearLayoutCompat
android:id="@+id/id_ll01"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center_horizontal"
android:orientation="vertical"
android:visibility="visible">
<ProgressBar
android:id="@+id/id_prog_01"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:layout_width="match_parent"
android:layout_height="16dp"
android:layout_marginHorizontal="28dp"
android:progressDrawable="@drawable/guide_pro" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginVertical="12dp"
android:text="Scanning the phone..."
android:textColor="@color/white"
android:textSize="15sp"
android:textStyle="bold" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginHorizontal="22dp"
android:layout_marginTop="4dp"
android:gravity="center"
android:text="To find unnecessary files completely, the first scan may take a moment."
android:textColor="#91E2FF"
android:textSize="13sp" />
</androidx.appcompat.widget.LinearLayoutCompat>
<androidx.appcompat.widget.LinearLayoutCompat
android:id="@+id/id_ll02"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center_horizontal"
android:orientation="vertical"
android:visibility="gone">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginVertical="12dp"
android:text="Click the button to clean the phone"
android:textColor="@color/white"
android:textSize="15sp"
android:textStyle="bold" />
<com.noober.background.view.BLTextView
android:id="@+id/id_clean_02"
android:layout_width="match_parent"
android:layout_height="50dp"
android:layout_marginHorizontal="50dp"
android:layout_marginTop="18dp"
android:gravity="center"
android:text="Clean"
android:textColor="@color/colorPrimary"
android:textSize="19sp"
android:textStyle="bold"
app:bl_corners_radius="90dp"
app:bl_solid_color="#6AFAE7" />
</androidx.appcompat.widget.LinearLayoutCompat>
</FrameLayout>
</androidx.constraintlayout.widget.ConstraintLayout>
\ No newline at end of file
{
"v": "5.7.6",
"fr": 8,
"ip": 0,
"op": 48,
"w": 1080,
"h": 1146,
"nm": "首次扫描",
"ddd": 0,
"assets": [
{
"id": "image_0",
"w": 798,
"h": 1068,
"u": "",
"p": "",
"e": 1
},
{
"id": "image_1",
"w": 780,
"h": 174,
"u": "",
"p": "",
"e": 1
},
{
"id": "image_2",
"w": 528,
"h": 918,
"u": "",
"p": "",
"e": 1
}
],
"layers": [
{
"ddd": 0,
"ind": 1,
"ty": 2,
"nm": "矩形 4077@3x.png",
"cl": "png",
"refId": "image_0",
"sr": 1,
"ks": {
"o": {
"a": 0,
"k": 100,
"ix": 11
},
"r": {
"a": 0,
"k": 0,
"ix": 10
},
"p": {
"a": 0,
"k": [
540,
573,
0
],
"ix": 2,
"l": 2
},
"a": {
"a": 0,
"k": [
399,
534,
0
],
"ix": 1,
"l": 2
},
"s": {
"a": 0,
"k": [
100,
100,
100
],
"ix": 6,
"l": 2
}
},
"ao": 0,
"ip": 0,
"op": 50,
"st": 0,
"bm": 0
},
{
"ddd": 0,
"ind": 2,
"ty": 2,
"nm": "组 4141@3x.png",
"cl": "png",
"refId": "image_1",
"sr": 1,
"ks": {
"o": {
"a": 0,
"k": 100,
"ix": 11
},
"r": {
"a": 0,
"k": 0,
"ix": 10
},
"p": {
"a": 1,
"k": [
{
"i": {
"x": 0.833,
"y": 0.833
},
"o": {
"x": 0.167,
"y": 0.167
},
"t": 0,
"s": [
540,
209,
0
],
"e": [
540,
1011,
0
],
"to": [
0,
0,
0
],
"ti": [
0,
0,
0
]
},
{
"i": {
"x": 0.833,
"y": 0.833
},
"o": {
"x": 0.167,
"y": 0.167
},
"t": 8,
"s": [
540,
1011,
0
],
"e": [
540,
209,
0
],
"to": [
0,
0,
0
],
"ti": [
0,
0,
0
]
},
{
"i": {
"x": 0.833,
"y": 0.833
},
"o": {
"x": 0.167,
"y": 0.167
},
"t": 16,
"s": [
540,
209,
0
],
"e": [
540,
1011,
0
],
"to": [
0,
0,
0
],
"ti": [
0,
0,
0
]
},
{
"i": {
"x": 0.833,
"y": 0.833
},
"o": {
"x": 0.167,
"y": 0.167
},
"t": 24,
"s": [
540,
1011,
0
],
"e": [
540,
209,
0
],
"to": [
0,
0,
0
],
"ti": [
0,
0,
0
]
},
{
"i": {
"x": 0.833,
"y": 0.833
},
"o": {
"x": 0.167,
"y": 0.167
},
"t": 32,
"s": [
540,
209,
0
],
"e": [
540,
1011,
0
],
"to": [
0,
0,
0
],
"ti": [
0,
0,
0
]
},
{
"i": {
"x": 0.833,
"y": 0.833
},
"o": {
"x": 0.167,
"y": 0.167
},
"t": 40,
"s": [
540,
1011,
0
],
"e": [
540,
209,
0
],
"to": [
0,
0,
0
],
"ti": [
0,
0,
0
]
},
{
"t": 48
}
],
"ix": 2,
"l": 2
},
"a": {
"a": 0,
"k": [
390,
87,
0
],
"ix": 1,
"l": 2
},
"s": {
"a": 0,
"k": [
100,
100,
100
],
"ix": 6,
"l": 2
}
},
"ao": 0,
"ip": 0,
"op": 50,
"st": 0,
"bm": 0
},
{
"ddd": 0,
"ind": 4,
"ty": 2,
"nm": "组 4160@3x.png",
"cl": "png",
"refId": "image_2",
"sr": 1,
"ks": {
"o": {
"a": 0,
"k": 100,
"ix": 11
},
"r": {
"a": 0,
"k": 0,
"ix": 10
},
"p": {
"a": 0,
"k": [
540,
583,
0
],
"ix": 2,
"l": 2
},
"a": {
"a": 0,
"k": [
264,
459,
0
],
"ix": 1,
"l": 2
},
"s": {
"a": 1,
"k": [
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 0,
"s": [
100,
100,
100
],
"e": [
96,
96,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 4,
"s": [
96,
96,
100
],
"e": [
100,
100,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 8,
"s": [
100,
100,
100
],
"e": [
96,
96,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 12,
"s": [
96,
96,
100
],
"e": [
100,
100,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 16,
"s": [
100,
100,
100
],
"e": [
96,
96,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 20,
"s": [
96,
96,
100
],
"e": [
100,
100,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 24,
"s": [
100,
100,
100
],
"e": [
96,
96,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 28,
"s": [
96,
96,
100
],
"e": [
100,
100,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 32,
"s": [
100,
100,
100
],
"e": [
96,
96,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 36,
"s": [
96,
96,
100
],
"e": [
100,
100,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 40,
"s": [
100,
100,
100
],
"e": [
96,
96,
100
]
},
{
"i": {
"x": [
0.833,
0.833,
0.833
],
"y": [
0.833,
0.833,
0.833
]
},
"o": {
"x": [
0.167,
0.167,
0.167
],
"y": [
0.167,
0.167,
0.167
]
},
"t": 44,
"s": [
96,
96,
100
],
"e": [
100,
100,
100
]
},
{
"t": 48
}
],
"ix": 6,
"l": 2
}
},
"ao": 0,
"ip": 0,
"op": 50,
"st": 0,
"bm": 0
}
],
"markers": []
}
\ No newline at end of file
This source diff could not be displayed because it is too large. You can view the blob instead.
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment