Delphi XE6 и Android(estimote beacon sdk) ClassCastException

Я занимался этим несколько дней без удачи. Это то, что java2pas обеспечивать BeaconManager$ServiceReadyCallback:

  [JavaSignature('com/estimote/sdk/BeaconManager$ServiceReadyCallback')]
  JBeaconManager_ServiceReadyCallback = interface(JObject)
    ['{335D3A89-7137-44CE-9969-BECC2F3AB8AC}']
    procedure onServiceReady ; cdecl;                                           // ()V A: $401
  end;

из этого я сделал анонимный метод следующим образом:

TAnServiceReadyCallBack = reference to function : JBeaconManager_ServiceReadyCallback;

ниже, как я называю это:

var
  an : TAnServiceReadyCallBack;
  bm : JBeaconManager;
begin
bm := TJBeaconManager.JavaClass.init(SharedActivityContext);
an := function : JBeaconManager_ServiceReadyCallback
      begin
         Result := TJBeaconManager_ServiceReadyCallback.Wrap((context as ILocalObject).GetObjectID);
      if Assigned(Result) then
         Memo1.Lines.Add('servicereadycallback init')
      else
        Memo1.Lines.Add('servicereadycallback not init')
end;
bm.connect(an); 

на этот вызов bm.connect(an) я получил следующее:

java.lang.ClassCastException: com.embarcadero.firemonkey.FMXNativeActivity cannot be cast to com.estimote.sdk.BeaconManager$ServiceReadyCallback

это то, что согласно оценке SDK:

   beaconManager.connect(new BeaconManager.ServiceReadyCallback() {
     @Override public void onServiceReady() {
       try {
         beaconManager.startRanging(ALL_ESTIMOTE_BEACONS);
       } catch (RemoteException e) {
         Log.e(TAG, "Cannot start ranging", e);
       }
     }
   });

код beacon sdk метод подключения выглядит следующим образом:

  public void connect(ServiceReadyCallback callback)
  {
    if (!checkPermissionsAndService()) {
      L.e("AndroidManifest.xml does not contain android.permission.BLUETOOTH or android.permission.BLUETOOTH_ADMIN permissions. BeaconService may be also not declared in AndroidManifest.xml.");
    }
    this.callback = ((ServiceReadyCallback)Preconditions.checkNotNull(callback, "callback cannot be null"));
    if (isConnectedToService()) {
      callback.onServiceReady();
    }
    boolean bound = this.context.bindService(new Intent(this.context, BeaconService.class), this.serviceConnection, 1);
    if (!bound) {
      L.w("Could not bind service: make sure thatcom.estimote.sdk.service.BeaconService is declared in AndroidManifest.xml");
    }
  }

Я также пытался сделать delphi Класс следующим образом:

  TServiceReadyCallback = class(TJavaLocal,
    JBeaconManager_ServiceReadyCallback)
  private
    fParent: TForm4;
  public
    constructor Create(parent: TForm4; context: JContext);
    function equals(o: JObject): boolean; cdecl;
    function getClass: Jlang_Class; cdecl;
    function hashCode: Integer; cdecl;
    procedure notify; cdecl;
    procedure notifyAll; cdecl;
    function toString: JString; cdecl;
    procedure wait; overload; cdecl;
    procedure wait(millis: Int64); overload; cdecl;
    procedure wait(millis: Int64; nanos: Integer); overload; cdecl;
    procedure onServiceReady; cdecl;
  end;

constructor TServiceReadyCallback.Create(parent: TForm4;
context: JContext);
begin
  fParent := parent;
  fParent.Memo1.Lines.Add('ServiceReadyCallback Created');
end;

используя этот класс, я использовал следующее:

var
  fservice: TServiceReadyCallback;
  context: JContext; 
  bm : JBeaconManager; 
begin
  CallInUIThreadAndWaitFinishing(
  procedure
  begin
  context := SharedActivityContext;
  bm := TJBeaconManager.JavaClass.init(context); 
  fservice := TServiceReadyCallback.Create(Self, context);
  bm.connect(fservice);
  end);
end;

при использовании этого я получил callback cannot be null' from beaconmanager.java линия

this.callback = ((ServiceReadyCallback)Preconditions.checkNotNull(callback, "callback cannot be null"));

Мне это очень нужно, так как мой клиент оказывает на меня давление. Я изучал это в течение по крайней мере 3-4 месяцев без какой-либо удачи. Кто-нибудь здесь, чтобы помочь мне решить эту проблему?

1 ответ

Вы не предоставили полное определение для оболочки JNI Bridge, которое должно выглядеть следующим образом:

  JBeaconManager_ServiceReadyCallbackClass = interface(JObjectClass)
    ['{D1DC624A-9431-4724-A0B6-2FE039BDCC33}']
    procedure onServiceReady ; cdecl;
  end;

  [JavaSignature('com/estimote/sdk/BeaconManager_ServiceReadyCallback')]
  JBeaconManager_ServiceReadyCallback = interface(JObject)
    ['{40394705-FCFB-442E-B3C7-3EA9825CDDBF}']
    procedure onServiceReady ; cdecl;
  end;

  TJBeaconManager_ServiceReadyCallback = class(TJavaGenericImport<JBeaconManager_ServiceReadyCallbackClass, JBeaconManager_ServiceReadyCallback>)
  end;
Другие вопросы по тегам