Amazon IAP не вызывает ответных вызовов слушателя (Kindle Fire, Xamarin)

Я пытаюсь [внедрить покупки в приложении Amazon в проекте Android с помощью Xamarin.

Настройка

Официальный Xamarin SDK от Amazon не поддерживает более новые 64-разрядные устройства, поэтому я создал собственную привязку библиотеки Java с помощью Amazon SDK от Java. ( Документация для версии Java)

Связывание, сгенерированное Xamarin для PurchasingService выглядит так:

namespace Com.Amazon.Device.Iap
{
    [Register ("com/amazon/device/iap/PurchasingService", DoNotGenerateAcw = true)]
    public sealed class PurchasingService : Object
    {
        //
        // Static Fields
        //
        private static IntPtr IS_SANDBOX_MODE_jfieldId;

        [Register ("SDK_VERSION")]
        public const string SdkVersion = "2.0.76.4";

        internal static IntPtr java_class_handle;

        private static IntPtr id_getUserData;

        private static IntPtr id_getProductData_Ljava_util_Set_;

        private static IntPtr id_getPurchaseUpdates_Z;

        private static IntPtr id_notifyFulfillment_Ljava_lang_String_Lcom_amazon_device_iap_model_FulfillmentResult_;

        private static IntPtr id_purchase_Ljava_lang_String_;

        private static IntPtr id_registerListener_Landroid_content_Context_Lcom_amazon_device_iap_PurchasingListener_;

        //
        // Static Properties
        //
        internal static IntPtr class_ref {
            get;
        }

        [Register ("IS_SANDBOX_MODE")]
        public static bool IsSandboxMode {
            get;
        }

        public static RequestId UserData {
            [Register ("getUserData", "()Lcom/amazon/device/iap/model/RequestId;", "GetGetUserDataHandler")]
            get;
        }

        //
        // Properties
        //
        protected override IntPtr ThresholdClass {
            get;
        }

        protected override Type ThresholdType {
            get;
        }

        //
        // Constructors
        //
        internal PurchasingService (IntPtr javaReference, JniHandleOwnership transfer);

        //
        // Static Methods
        //
        [Register ("getProductData", "(Ljava/util/Set;)Lcom/amazon/device/iap/model/RequestId;", "")]
        public static RequestId GetProductData (ICollection<string> p0);

        [Register ("getPurchaseUpdates", "(Z)Lcom/amazon/device/iap/model/RequestId;", "")]
        public static RequestId GetPurchaseUpdates (bool p0);

        [Register ("notifyFulfillment", "(Ljava/lang/String;Lcom/amazon/device/iap/model/FulfillmentResult;)V", "")]
        public static void NotifyFulfillment (string p0, FulfillmentResult p1);

        [Register ("purchase", "(Ljava/lang/String;)Lcom/amazon/device/iap/model/RequestId;", "")]
        public static RequestId Purchase (string p0);

        [Register ("registerListener", "(Landroid/content/Context;Lcom/amazon/device/iap/PurchasingListener;)V", "")]
        public static void RegisterListener (Context p0, IPurchasingListener p1);
    }
}

И для PurchasingListener интерфейс:

namespace Com.Amazon.Device.Iap
{
    [Register ("com/amazon/device/iap/PurchasingListener", "", "Com.Amazon.Device.Iap.IPurchasingListenerInvoker")]
    public interface IPurchasingListener : IJavaObject, IDisposable
    {
        //
        // Methods
        //
        [Register ("onProductDataResponse", "(Lcom/amazon/device/iap/model/ProductDataResponse;)V", "GetOnProductDataResponse_Lcom_amazon_device_iap_model_ProductDataResponse_Handler:Com.Amazon.Device.Iap.IPurchasingListenerInvoker, AmazonIAP")]
        void OnProductDataResponse (ProductDataResponse p0);

        [Register ("onPurchaseResponse", "(Lcom/amazon/device/iap/model/PurchaseResponse;)V", "GetOnPurchaseResponse_Lcom_amazon_device_iap_model_PurchaseResponse_Handler:Com.Amazon.Device.Iap.IPurchasingListenerInvoker, AmazonIAP")]
        void OnPurchaseResponse (PurchaseResponse p0);

        [Register ("onPurchaseUpdatesResponse", "(Lcom/amazon/device/iap/model/PurchaseUpdatesResponse;)V", "GetOnPurchaseUpdatesResponse_Lcom_amazon_device_iap_model_PurchaseUpdatesResponse_Handler:Com.Amazon.Device.Iap.IPurchasingListenerInvoker, AmazonIAP")]
        void OnPurchaseUpdatesResponse (PurchaseUpdatesResponse p0);

        [Register ("onUserDataResponse", "(Lcom/amazon/device/iap/model/UserDataResponse;)V", "GetOnUserDataResponse_Lcom_amazon_device_iap_model_UserDataResponse_Handler:Com.Amazon.Device.Iap.IPurchasingListenerInvoker, AmazonIAP")]
        void OnUserDataResponse (UserDataResponse p0);
    }
}

Я также убедился, что манифест Android был изменен с помощью:

<receiver android:name = "com.amazon.device.iap.ResponseReceiver" >
    <intent-filter>
        <action android:name = "com.amazon.inapp.purchasing.NOTIFY"
          android:permission = "com.amazon.inapp.purchasing.Permission.NOTIFY" />
    </intent-filter>
</receiver>

Amazon App Tester установлен на моем Kindle и настроен со следующими тестовыми данными:

{
    "IAP_ID":
    {
      "description":"One year Subscription",
      "title":"My Subscription",
      "itemType":"SUBSCRIPTION",
      "price":20.00,
      "subscriptionParent":"org.company.appid"
    }
}

И у меня есть скелет класса, чтобы проверить, что система работает:

public class AmazonIAPController : Java.Lang.Object, IPurchasingListener
{
    const string ProductId = "IAP_ID";
    private Activity uiActivity;

    public AmazonIAPController()
    {
    }

    public void SetupIAP(Activity activity) {
        uiActivity = activity;
        PurchasingService.RegisterListener(uiActivity, this);

        // test that the Amazon system works
        PurchasingService.Purchase(ProductId);
    }

    public void OnResume(Activity activity) {
        uiActivity = activity;
        PurchasingService.RegisterListener(uiActivity, this);

        //PurchasingService.UserData;
        PurchasingService.GetPurchaseUpdates(false);
        PurchasingService.GetProductData(new string[] { ProductId });

    }

    public void OnProductDataResponse(ProductDataResponse response) {
        var status = response.GetRequestStatus();

        Debug.WriteLine("*****OnProductDataResponse");
    }

    public void OnPurchaseResponse(PurchaseResponse response) {
        var status = response.GetRequestStatus();
        Debug.WriteLine("*****OnPurchaseResponse");
    }

    public void OnPurchaseUpdatesResponse(PurchaseUpdatesResponse response) {
        var status = response.GetRequestStatus();
        Debug.WriteLine("*****OnPurchaseUpdatesResponse");
    }

    public void OnUserDataResponse(UserDataResponse response) {
        var status = response.GetRequestStatus();
        Debug.WriteLine("*****OnUserDataResponse");
    }
}

Тестовый прогон

Я пытаюсь запустить это на своем 5-м Gen Kindle Fire. Когда приложение запускается, пользователю предлагается приглашение Amazon IAP, как и ожидалось. Если покупка совершена, она проходит, и транзакция правильно отображается в приложении Amazon App Tester.

Эта проблема

У меня есть настройки точек останова в методах обратного вызова. Эти точки останова никогда не срабатывают. Ни один не делает Debug.WriteLine вызовы отображаются в приложении вывода. Похоже, что мои методы обратного вызова не вызываются.

1 ответ

Решение

Корень проблемы в том, что записи манифеста Android конфликтуют с тем, как Xamarin генерирует записи манифеста с атрибутами кода.

Чтобы устранить проблему, я удалил вышеупомянутые записи из манифеста и создал файл дополнения в проекте привязки Java, содержащий частичный класс ResponseReciever и соответствующие атрибуты манифеста следующим образом:

using System;
using Android.App;
using Android.Content;

namespace Com.Amazon.Device.Iap
{
    [BroadcastReceiver(Name = "com.amazon.device.iap.ResponseReceiver")]
    [IntentFilter(new[] { "com.amazon.inapp.purchasing.NOTIFY" })]
    public partial class ResponseReceiver
    {

    }
}

После этого записи манифеста были сгенерированы правильно, и ожидаемые обратные вызовы начали работать.

Замечания:

Вы, вероятно, заметите, что android:permission запись из манифеста XML отсутствует в указанных выше атрибутах. Я не могу найти ссылку, но я помню, что где-то видел, что не только эта запись не нужна для работы SDK, но на самом деле это недопустимый атрибут action согласно официальной документации Android.

Другие вопросы по тегам