アプリ内課金のバージョン3の「TrivialDrive」の例に基づいた手順全体に基づいて、テスト目的でアプリ内課金をアプリに含めようとしました(そして、IABファイルの変更されていないバージョンを実装します「util」サブディレクトリのデモ)、しかしそれは私にとっては機能しません-LogCatでは、アプリがエラーで終了する直前に、「アプリ内課金エラー:操作の不正な状態( launchPurchaseFlow):IAB Helperが設定されていません。 "(startRegistered()関数が起動され、「登録ボタンがクリックされました。アップグレードの購入フローを起動しています。」というログメッセージが表示された直後))...
ここで何がうまくいかないのか?
コードの関連部分は次のとおりです。
package com.mytest;
(..)
import com.mytest.iab.IabHelper; // the originals from the demo example, unmodified
import com.mytest.iab.IabResult;
import com.mytest.iab.Inventory;
import com.mytest.iab.Purchase;
public class Result3 extends Activity implements OnClickListener {
private static final String TAG = "BillingService";
private Context mContext;
boolean mIsRegistered = false;
// this has already been set up for my app at the publisher's console
static final String IS_REGISTERED = "myregistered";
static final int RC_REQUEST = 10001;
// The helper object
IabHelper mHelper;
/** Call when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.result3);
mContext = this;
String base64EncodedPublicKey = "[my public key]"; // (from publisher's console for my app)
// Create the helper, passing it our context and the public key to verify signatures with
Log.d(TAG, "Creating IAB helper.");
mHelper = new IabHelper(this, base64EncodedPublicKey);
// enable debug logging (for a production application, you should set this to false).
mHelper.enableDebugLogging(true);
// Start setup. This is asynchronous and the specified listener
// will be called once setup completes.
Log.d(TAG, "Starting setup.");
mHelper.startSetup(new IabHelper.OnIabSetupFinishedListener() {
public void onIabSetupFinished(IabResult result) {
Log.d(TAG, "Setup finished.");
if (!result.isSuccess()) {
complain("Problem setting up in-app billing: " + result);
return;
}
// Hooray, IAB is fully set up. Now, let's get an inventory of stuff we own.
Log.d(TAG, "Setup successful. Querying inventory.");
mHelper.queryInventoryAsync(mGotInventoryListener);
}
});
// Set the onClick listeners
findViewById(R.id.btnPurchase).setOnClickListener(this);
}
// Listener that's called when we finish querying the items we own
IabHelper.QueryInventoryFinishedListener mGotInventoryListener = new IabHelper.QueryInventoryFinishedListener() {
public void onQueryInventoryFinished(IabResult result, Inventory inventory) {
Log.d(TAG, "Query inventory finished.");
if (result.isFailure()) {
complain("Failed to query inventory: " + result);
return;
}
Log.d(TAG, "Query inventory was successful.");
// Do we have the premium upgrade?
mIsRegistered = inventory.hasPurchase(IS_REGISTERED);
Log.d(TAG, "User is " + (mIsRegistered ? "REGISTERED" : "NOT REGISTERED"));
setWaitScreen(false);
Log.d(TAG, "Initial inventory query finished; enabling main UI.");
}
};
// User clicked the "Register" button.
private void startRegistered() {
Log.d(TAG, "Register button clicked; launching purchase flow for upgrade.");
setWaitScreen(true);
mHelper.launchPurchaseFlow(this, IS_REGISTERED, RC_REQUEST, mPurchaseFinishedListener);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
Log.d(TAG, "onActivityResult(" + requestCode + "," + resultCode + "," + data);
// Pass on the activity result to the helper for handling
if (!mHelper.handleActivityResult(requestCode, resultCode, data)) {
// not handled, so handle it ourselves (here's where you'd
// perform any handling of activity results not related to in-app billing..
super.onActivityResult(requestCode, resultCode, data);
}
else {
Log.d(TAG, "onActivityResult handled by IABUtil.");
}
}
// Callback for when a purchase is finished
IabHelper.OnIabPurchaseFinishedListener mPurchaseFinishedListener = new IabHelper.OnIabPurchaseFinishedListener() {
public void onIabPurchaseFinished(IabResult result, Purchase purchase) {
Log.d(TAG, "Purchase finished: " + result + ", purchase: " + purchase);
if (result.isFailure()) {
// Oh noes!
complain("Error purchasing: " + result);
setWaitScreen(false);
return;
}
Log.d(TAG, "Purchase successful.");
if (purchase.getSku().equals(IS_REGISTERED)) {
Log.d(TAG, "User has registered..");
alert("Thank you.");
mIsRegistered = true;
setWaitScreen(false);
}
}
};
// We're being destroyed. It's important to dispose of the helper here!
@Override
public void onDestroy() {
// very important:
Log.d(TAG, "Destroying helper.");
if (mHelper != null) mHelper.dispose();
mHelper = null;
}
void complain(String message) {
Log.e(TAG, "**** Register Error: " + message);
alert("Error: " + message);
}
void setWaitScreen(boolean set) {
// just a dummy for now
}
void alert(String message) {
AlertDialog.Builder bld = new AlertDialog.Builder(this);
bld.setMessage(message);
bld.setNeutralButton("OK", null);
Log.d(TAG, "Showing alert dialog: " + message);
bld.create().show();
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.btnPurchase:
startRegistered();
break;
default:
break;
}
}
}
Logcatのその他の行:
12-20 01:06:36.701: D/dalvikvm(299): GC_FOR_MALLOC freed 4262 objects / 308592 bytes in 84ms
12-20 01:06:36.701: D/webviewglue(299): nativeDestroy view: 0x2ea718
12-20 01:06:36.771: W/webcore(299): Can't get the viewWidth after the first layout
12-20 01:07:07.111: W/webcore(299): Can't get the viewWidth after the first layout
12-20 01:07:18.510: D/webviewglue(299): nativeDestroy view: 0x2dd458
12-20 01:07:18.510: D/dalvikvm(299): GC_FOR_MALLOC freed 6042 objects / 544504 bytes in 50ms
12-20 01:07:18.530: D/webviewglue(299): nativeDestroy view: 0x2ea8d0
12-20 01:07:18.660: D/BillingService(299): Creating IAB helper.
12-20 01:07:18.660: D/BillingService(299): Starting setup.
12-20 01:07:18.660: D/IabHelper(299): Starting in-app billing setup.
12-20 01:07:19.621: W/webcore(299): Can't get the viewWidth after the first layout
12-20 01:07:20.160: W/webcore(299): Can't get the viewWidth after the first layout
12-20 01:07:32.481: D/webviewglue(299): nativeDestroy view: 0x3f88e8
12-20 01:07:32.491: D/dalvikvm(299): GC_FOR_MALLOC freed 5798 objects / 513640 bytes in 50ms
12-20 01:07:32.511: D/BillingService(299): Register button clicked; launching purchase flow for upgrade.
12-20 01:07:32.511: E/IabHelper(299): In-app billing error: Illegal state for operation (launchPurchaseFlow): IAB helper is not set up.
12-20 01:07:32.521: D/AndroidRuntime(299): Shutting down VM
12-20 01:07:32.521: W/dalvikvm(299): threadid=1: thread exiting with uncaught exception (group=0x4001d800)
12-20 01:07:32.541: E/AndroidRuntime(299): FATAL EXCEPTION: main
12-20 01:07:32.541: E/AndroidRuntime(299): Java.lang.IllegalStateException: IAB helper is not set up. Can't perform operation: launchPurchaseFlow
12-20 01:07:32.541: E/AndroidRuntime(299): at com.test_ed.iab.IabHelper.checkSetupDone(IabHelper.Java:673)
12-20 01:07:32.541: E/AndroidRuntime(299): at com.test_ed.iab.IabHelper.launchPurchaseFlow(IabHelper.Java:315)
12-20 01:07:32.541: E/AndroidRuntime(299): at com.test_ed.iab.IabHelper.launchPurchaseFlow(IabHelper.Java:294)
12-20 01:07:32.541: E/AndroidRuntime(299): at com.test_ed.Result3.startRegistered(Result3.Java:157)
12-20 01:07:32.541: E/AndroidRuntime(299): at com.test_ed.Result3.onClick(Result3.Java:248)
12-20 01:07:32.541: E/AndroidRuntime(299): at Android.view.View.performClick(View.Java:2408)
12-20 01:07:32.541: E/AndroidRuntime(299): at Android.view.View$PerformClick.run(View.Java:8816)
12-20 01:07:32.541: E/AndroidRuntime(299): at Android.os.Handler.handleCallback(Handler.Java:587)
12-20 01:07:32.541: E/AndroidRuntime(299): at Android.os.Handler.dispatchMessage(Handler.Java:92)
12-20 01:07:32.541: E/AndroidRuntime(299): at Android.os.Looper.loop(Looper.Java:123)
12-20 01:07:32.541: E/AndroidRuntime(299): at Android.app.ActivityThread.main(ActivityThread.Java:4627)
12-20 01:07:32.541: E/AndroidRuntime(299): at Java.lang.reflect.Method.invokeNative(Native Method)
12-20 01:07:32.541: E/AndroidRuntime(299): at Java.lang.reflect.Method.invoke(Method.Java:521)
12-20 01:07:32.541: E/AndroidRuntime(299): at com.Android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.Java:868)
12-20 01:07:32.541: E/AndroidRuntime(299): at com.Android.internal.os.ZygoteInit.main(ZygoteInit.Java:626)
12-20 01:07:32.541: E/AndroidRuntime(299): at dalvik.system.NativeStart.main(Native Method)
PurchaseFlow関数の実行中に同じ問題が発生しました。 Googleの例のActivityクラス、特にメソッドprotected void onActivityResult(int requestCode, int resultCode, Intent data)
を見てください。おそらくこれを実装するのを忘れていました。この機能は、メカニズム全体がグリッチなしで機能するために不可欠です。
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
Log.i(TAG, "onActivityResult(" + requestCode + "," + resultCode + "," + data);
// Pass on the activity result to the helper for handling
if (!inappBillingHelper.handleActivityResult(requestCode, resultCode, data)) {
super.onActivityResult(requestCode, resultCode, data);
}
else {
Log.i(TAG, "onActivityResult handled by IABUtil.");
}
}
編集:さらに、携帯電話のGmailアカウントに間違ったパスワードが関連付けられている場合にも問題が発生します(これは今日私に起こりました)。もちろん、すべてのInapp請求機能は電話でテストする必要がありますが、それは明らかだと思います。
fundamentalの問題は、startRegistered()がUIユーザーのクリックに直接応答して呼び出されるのに対して、IabHelperオブジェクトのセットアップが非同期でトリガーされるため、非同期応答が完了するまで完了したことがわからないことです。 onIabSetupFinished()を介して受信されます。
StartRegistered()メソッドは、ユーザーのクリックによってトリガーされ、launchPurchaseFlow()を呼び出します。これは、IabHelperオブジェクトが既にセットアップを完了していることを必要としますが、ユーザーがクリックして、失敗したか、ユーザーが非常に迅速に描画を行うため)、セットアップは完了せず、launchPurchaseFlow()は表示されているエラーを報告します。 logcatの場合、遅延は14秒で、通常は十分な時間ですが、この場合はそうではないかもしれません。または、何かがうまくいかなかったかもしれず、あなたがどれだけ長く待っていても接続することはなかっただろう。
Logcatには、「請求サービスが接続されました」というメッセージはありません。これは、セットアップを完了する場合に最初に発生しなければならないことの1つです。これは発生しないため、onIabSetupFinished()からのメッセージ(成功または失敗のいずれか)も表示されません。
非同期応答が必要なため、これは扱いにくいものです。 1つのアプローチは、onIabSetupFinished()が成功して戻るまで購入をトリガーするために使用されるボタンを無効にすることです。これにより、IabHelperオブジェクトが正常にセットアップされるまで、購入がトリガーされなくなります。もちろん、セットアップが失敗した場合、機能しないボタンが表示されますが、少なくともセットアップの完了を待っていることを示すメッセージを表示することで、ユーザーに何が起きているかを伝えることができます。ボタンのテキスト)。
それでも、購入が開始されて支払いダイアログがユーザーに表示されると、ユーザーが購入を検討している間にアプリがメモリからフラッシュされるonStop()サイクルをアプリが通過する可能性があります(購入ダイアログは一部であるため)アプリの一部ではないGoogle PlayそれはIabHelper()オブジェクトを破壊し、それから作成し、非同期で設定する必要がありますagain。また、これはonCreate()メソッドで非同期にトリガーされるため、onActivityResult()はGoogle Play開発者サービスによって呼び出され、ユーザーの購入アクションを報告しますbefore IabHelperオブジェクトのセットアップが完了し、 onActivityResult()では、se IabHelperインスタンスを使用する必要があります。これによりエラーが発生する可能性があります。何かに備えなければならないようです。
これはあなたがあなたが扱っているものの風味を与えるはずです。 IABはまさにこれらの理由で困難です-非同期のものの複数のスレッド(例:セットアップvs.購入vs. Android使用するためにアプリを停止して、おそらくアプリが購入の結果を取得するのを待っているGoogle Playアプリの購入操作)(TrivialDriveサンプルを含む)実装されるものの多くは、実際にはアプリがメモリ内に留まることに暗黙的に依存しているため、不安定ですリサイクルされるか、競合状態の一方のレッグ(セットアップなど)が完了してから、別のレッグ(購入の打ち上げなど)が完了するなど。
まったく同じ問題に頭を包み込みました。 IabHelper-Setupが開始されますが、その後は何も起こりません。また、アプリ内購入をクリックすると、まったく同じエラーが返されます。
私が理解したのは、Eclipseのエミュレーターのみを使用したことです。特定のGoogle Playバージョンが必要であることを読んだ後、テストエミュレーションドライブにGoogle Playが完全に欠けていることに気付きました。
その後、実際の電話を使用したところ、問題なく動作しました!したがって、まだその問題が解決しない場合は、実際のデバイスを使用してみてください(使用可能なデバイスがある場合)。これでうまくいくはずです。
私が出会ったもう一つのこと。アプリ内課金の最新バージョンをサポートするGoogle Playの最新バージョンがデバイス上にあるかもしれませんが、他のユーザーはそうではないかもしれません。そして、これにより理論的にはクラッシュが開発者コンソールに表示されるはずですが、firebaseを実装するまでこれらのクラッシュを見ることができませんでした...そして、私はそれらの多くを見ました。私がやったことは、try catchを使用して、Google Playの最新バージョンを持っていないか、Google Playストアで問題が発生したユーザーをこのページにリンクすることです https://support.google.com/googleplay/answer/1050566?hl = en
try {
mHelper.launchPurchaseFlow(this, SKU_PRO_LT, RC_REQUEST,
mPurchaseFinishedListener, payload);
} catch (Exception e) { //with IabHelper.IabAsyncInProgressException the code still fatally crashes for some reason
//complain("Error launching purchase flow. Another async operation in progress.");
alert2("[error msg]");
setWaitScreen(false);
}
alert2は、上記のWebページへのリンクを含む単なるダイアログボックスです。
ただし、最初に、コードの問題ではなくPlayストアの更新の問題であることを確認するために、友人の携帯電話でアプリを購入してテストすることをお勧めします。