Вот полная реализация.
Для получения дополнительной информации см. Документацию: https://developer.android.com/google/play/billing/billing_library_overview
String ITEM_SKU_diamond_500 = "diamond_500";
BillingClient billingClient;
AcknowledgePurchaseResponseListener acknowledgePurchaseResponseListener;
String premiumUpgradePrice = "";
1: клиент биллинга должен быть создан с использованием BillingClient.Builder
.
billingClient = BillingClient.newBuilder(this)
.enablePendingPurchases()
.setListener(this).build();
2: После создания billingClient запустите соединение billingClient
billingClient.startConnection(new BillingClientStateListener() {
@Override
public void onBillingSetupFinished(BillingResult billingResult) {
if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK) {
List<String> skuList = new ArrayList<>();
skuList.add(ITEM_SKU_diamond_500);
final SkuDetailsParams.Builder params = SkuDetailsParams.newBuilder();
params.setSkusList(skuList).setType(BillingClient.SkuType.INAPP);
3: После установления успешного соединения вызовите метод querySkuDetailsAsyn c billingClient для асинхронной выборки деталей 'sku'.
billingClient.querySkuDetailsAsync(params.build(), new SkuDetailsResponseListener() {
@Override
public void onSkuDetailsResponse(BillingResult billingResult, List<SkuDetails> skuDetailsList) {
if (skuDetailsList != null && billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK) {
for (SkuDetails skuDetails : skuDetailsList) {
String sku = skuDetails.getSku();
String price = skuDetails.getPrice();
final BillingFlowParams params = BillingFlowParams.newBuilder()
.setSkuDetails(skuDetails)
.build();
if (ITEM_SKU_diamond_500.equals(sku)) {
premiumUpgradePrice = price;
firstBtn500(params);
}
}
} else if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.ERROR) {
Toast.makeText(DiamondsActivity.this, "Error", Toast.LENGTH_SHORT).show();
}
}
});
} else if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.SERVICE_TIMEOUT) {
Toast.makeText(DiamondsActivity.this, "Service timeout", Toast.LENGTH_SHORT).show();
} else {
Toast.makeText(DiamondsActivity.this, "Failed to connect to the billing client", Toast.LENGTH_SHORT).show();
}
}
@Override
public void onBillingServiceDisconnected() {
Toast.makeText(DiamondsActivity.this, "Disconnected from the client", Toast.LENGTH_SHORT).show();
}
});
acknowledgePurchaseResponseListener = new AcknowledgePurchaseResponseListener() {
@Override
public void onAcknowledgePurchaseResponse(BillingResult billingResult) {
Toast.makeText(DiamondsActivity.this, "Purchase acknowledged", Toast.LENGTH_SHORT).show();
}
};
4: Когда пользователь пытается приобрести продукт в приложении или подписке, проверьте, поддерживается ли продукт с помощью метода billngClient's isFeatureSupported(BillingClient.FeatureType
. / * SUBSCRIPTIONS или другого * /), и вызовите метод launchBillingFlow для billingClient.
private void firstBtn500(final BillingFlowParams params) {
firstPurchaseBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
billingClient.launchBillingFlow(DiamondsActivity.this, params);
}
});
}
ОБНОВЛЕНИЕ
Здесь вы можете проверить, если товар уже приобретен или нет
@Override
public void onPurchasesUpdated(BillingResult billingResult, @Nullable List<Purchase> purchases) {
if (purchases != null && billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK) {
for (Purchase purchase : purchases) {
handlePurchases(purchase);
}
} else if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.USER_CANCELED) {
Toast.makeText(this, "Purchased Canceled", Toast.LENGTH_SHORT).show();
}
else if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.ITEM_ALREADY_OWNED) {
Toast.makeText(this, "Already Purchased", Toast.LENGTH_SHORT).show();
}
}
Если вы хотите приобрести товар один раз, вам нужно Acknowledge
приобрести
private void handlePurchases(final Purchase purchase) {
if (purchase.getPurchaseState() == Purchase.PurchaseState.PURCHASED) {
//Acknowledge the purchase if it hasn't already been acknowledged.
if (!purchase.isAcknowledged()) {
AcknowledgePurchaseParams acknowledgePurchaseParams =
AcknowledgePurchaseParams.newBuilder()
.setPurchaseToken(purchase.getPurchaseToken())
.build();
billingClient.acknowledgePurchase(acknowledgePurchaseParams, acknowledgePurchaseResponseListener);
}
Если вы хотите покупать один и тот же предмет снова и снова (например, в приложении / игре в монетах или в кредит или что-то в этом роде). Вы должны потреблять покупки ниже код для потребления покупки.
// Todo :Consume the purchase async
ConsumeParams consumeParams = ConsumeParams.newBuilder()
.setPurchaseToken(purchase.getPurchaseToken())
.build();
ConsumeResponseListener consumeResponseListener = new ConsumeResponseListener() {
@Override
public void onConsumeResponse(BillingResult billingResult, String purchaseToken) {
Toast.makeText(DiamondsActivity.this, "Purchase successful", Toast.LENGTH_SHORT).show();
if (billingResult.getResponseCode() == BillingClient.BillingResponseCode.OK) {
if (purchase.getSku().equalsIgnoreCase(ITEM_SKU_diamond_500)) {
Toast.makeText(DiamondsActivity.this, "Thank you for purchasing!", Toast.LENGTH_SHORT).show();
}
}
}
};
billingClient.consumeAsync(consumeParams, consumeResponseListener);
} else if (purchase.getPurchaseState() == Purchase.PurchaseState.PENDING) {
Toast.makeText(this, "Purchase pending", Toast.LENGTH_SHORT).show();
}
}