- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
由于最近对 Google Billing 库和开发者控制台进行了所有更改,因此很难应用 2014 年的答案,因此我再次发帖讨论这个主题,希望能找到一个现代的答案。
我正在使用 Google 结算库 2.2.0
相关代码示例:
private final String SKU_TEST_PROD_1YR = "test_prod_id_1_year";
private final String SKU_TEST_PROD_6MN = "test_prod_id_6_month";
private final String SKU_TEST_PROD_1MN = "test_prod_id_1_month";
private BillingClient billingClient;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
List<String> iapProdIdList = Arrays.asList(SKU_TEST_PROD_1YR, SKU_TEST_PROD_6MN, SKU_TEST_PROD_1MN);
setupBillingClient(this, iapProdIdList);
}
@Override
protected void onDestroy() {
super.onDestroy();
billingClient.endConnection();
}
private void setupBillingClient(Context context, List<String> iapProdList) {
billingClient = BillingClient.newBuilder(context).setListener(this).build();
billingClient.startConnection(new BillingClientStateListener() {
@Override
public void onBillingSetupFinished(BillingResult billingResult) {
if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK) {
loadAllSubscriptionProductIdDetails(iapProdList);
}
}
@Override
public void onBillingServiceDisconnected() { }
});
}
private void loadAllSubscriptionProductIdDetails(List<String> iapProdList) {
if (billingClient.isReady()) {
SkuDetailsParams params = SkuDetailsParams.newBuilder().setSkusList(iapProdList).setType(BillingClient.SkuType.SUBS).build();
billingClient.querySkuDetailsAsync(params, (billingResult, prodDetailList) -> {
if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK && !prodDetailList.isEmpty()) {
for (SkuDetails prodDetail : prodDetailList) {
String productSku = prodDetail.getSku();
switch(productSku) {
case SKU_TEST_PROD_1YR : {
final BillingFlowParams billingFlowParams = BillingFlowParams.newBuilder().setSkuDetails(prodDetail).build();
buttonIap1Yr.setOnClickListener(v -> {
// trigger purchase - Google needs the parent activity to overlay with their UI
billingClient.launchBillingFlow(this, billingFlowParams);
});
break;
}
case SKU_TEST_PROD_6MN : {
final BillingFlowParams billingFlowParams = BillingFlowParams.newBuilder().setSkuDetails(prodDetail).build();
buttonIap6mth.setOnClickListener(v -> {
// trigger purchase - Google needs the parent activity to overlay with their UI
billingClient.launchBillingFlow(this, billingFlowParams);
});
break;
}
case SKU_TEST_PROD_1MN : {
final BillingFlowParams billingFlowParams = BillingFlowParams.newBuilder().setSkuDetails(prodDetail).build();
buttonIap1mth.setOnClickListener(v -> {
// trigger purchase - Google needs the parent activity to overlay with their UI
billingClient.launchBillingFlow(this, billingFlowParams);
});
break;
}
default :
Toast.makeText(this, "Did not find Product in-app: "+ productSku, Toast.LENGTH_SHORT).show();
}
}//end of FOR
}//end of IF
});
} else {
Toast.makeText(this, "billingClient is NOT ready", Toast.LENGTH_SHORT).show();
}
}
@Override
public void onPurchasesUpdated(BillingResult billingResult, List<Purchase> list) {
if (billingResult != null && list != null) {
if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK) {
for (Purchase purchase : list) {
if(!purchase.isAcknowledged()) acknowledgePurchase(purchase.getPurchaseToken());
}
}
else if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.USER_CANCELED) {
Toast.makeText(this, "User Canceled", Toast.LENGTH_SHORT).show();
}
else if(billingResult.getResponseCode()== BillingClient.BillingResponseCode.ITEM_ALREADY_OWNED) {
Toast.makeText(this, "Already Purchased", Toast.LENGTH_SHORT).show();
}
}
}
private void acknowledgePurchase(String purchaseToken) {
AcknowledgePurchaseParams params = AcknowledgePurchaseParams.newBuilder().setPurchaseToken(purchaseToken).build();
billingClient.acknowledgePurchase(params, billingResult -> {
if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK) {
Toast.makeText(this, "Purchase Acknowledged", Toast.LENGTH_SHORT).show();
}
});
}
当我点击任何 [购买订阅] 按钮时,billingClient.launchBillingFlow(this, billingFlowParams);
被正确触发并且 SKU 正确。
据我所知,所有开发控制台都已正确设置。我想向我的经理演示 IAP,但是,“无法找到您尝试购买的商品”让我的风格受到限制!我做错了什么?
最佳答案
我在问题中发布的代码功能齐全。问题是 100% 由于开发者控制台配置:
修复它:
1) 最低限度将应用提升到 alpha 版(内部测试轨道不起作用!)
2) 单击 Alpha 轨道上的 [管理] 并将选择加入链接通过电子邮件发送给您的测试人员
3) 您的测试人员会说“当我点击此链接时,Play 商店说找不到该应用程序”。告诉他们放松 30 分钟 - Google 控制台需要 30 分钟来处理这种复杂情况。
让他们下载早期发布版本,IAP 突然就可以使用了!
关于安卓计费;如何解决 "The item you were attempting to purchase could not be found",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61090660/
尝试从对话框中的 EditText 获取 Edit Text 的值,但一次又一次地出现此错误 Attempt to invoke virtual method 'android.text.Editab
最近尝试了一下最新的Laravel(6.4)。正在尝试实现基于 API 的简单登录功能。没有使用 Passport 或 Tymon 的 JWT 等任何软件包。我使用了非常基本的身份验证(只需在用户表中
最近尝试了一下最新的Laravel(6.4)。正在尝试实现基于 API 的简单登录功能。没有使用 Passport 或 Tymon 的 JWT 等任何软件包。我使用了非常基本的身份验证(只需在用户表中
我在我的 PC 上运行 cifar10 网络,在完成训练和运行评估脚本后出现以下错误: 2016-06-01 14:37:14.238317: precision @ 1 = 0.000 Traceb
我正在使用 ng2-toastr 并收到以下错误 https://www.npmjs.com/package/ng2-toastr Attempt to use a destroyed view: d
env file:环境文件: APP_ENV=localAPP_DEBUG=trueAPP_KEY= ...........DB_HOST=srv3.linuxisrael.co.ilDB_D
当我登录管理员时,我正在尝试对 api 进行多重保护身份验证,我得到了跟随错误 BadMethodCallException Method Illuminate\Auth\Req uestGuard:
我开始在 Lua 中进行编程,并在运行脚本时出现此错误: attempt to index upvalue 'base' (a function value) 这可能是因为我还没有掌握一些非常基本的东
我试图在 Oozie 工作流中聚合一些数据。但是聚合步骤失败。 我在日志中发现了两个兴趣点:第一个是一个似乎重复出现的错误(?): 容器完成后,它会被杀死,但会以非零退出代码 143 退出。 它完成了
我的问题是当我调用函数时: [self performSegueWithIdentifier: @"FinalPlayPT" sender: self]; 它有效,但我有这个警告: Warning:
项目背景 项目整体采用的是springboot+mybatis 方式。有一次做数据查询的时候。console突然报:DataIntegrityViolationException: Error att
我在使用 Jobs 发送电子邮件的 Ubuntu Server 上设置了 Laravel 项目。 下面是我在 中的 laravel-worker 文件/etc/supervisor/laravel-w
尝试运行我的 React 应用程序时收到以下错误: ./src/components/App/App.js Attempted import error: 'combineReducers' is n
我的编码功能是这样的: fn encode_login(packet: &str) { let bytes = packet.as_bytes(); for (i, &element)
如果一个版本号大于另一个版本号,我的 msbuild 需要采取有条件的操作。我尝试像下面这样编写代码,但发现出现错误(也在下面)。我哪里出错了?
我是 Lua 的新手,并试图让事情在我的脑海中进行排序。我试过这个代码: function newCarousel(images) local slideToImage = function(
我正在使用 entrust用于在 Laravel 5.3 中管理基于角色的权限,并自然地为不同的用户类型使用手动/自定义登录。 Auth::attempt()可以吗?处理外表关系?基本上,我想做这样的
我谷歌了一遍又一遍,但没有对我有用的信息。 情况是这样的:1.产品有 active 。2.我通过测试账号来测试我的apk。3.我已经上传了一整天的apk。 但是当我想为应用程序中的项目付款时,goog
我正在尝试使用 groovy 解析日期字符串,但遇到了问题。这是字符串的样子和不起作用的逻辑。 def dateString = "2017-01-01T12:00:00Z" def date = D
我正在使用 ParseFacebookUtils 从 Facebook 登录我的应用。 LoginActivity 的 onCreate: protected void onCreate(Bundle
我是一名优秀的程序员,十分优秀!