Оценка обнаружения области маяков из службы в фоновом режиме

Недавно я начал тестировать маяки Estimote и пытаюсь запустить уведомление из фоновой службы при входе в регион маяка, но, к сожалению, мое решение не работает. Ошибок не выдает, но уведомление не запускается при обнаружении маяка. Я не знаю, является ли это какой-то ошибкой кода или просто способ сделать это неправильно. Я прочитал этот другой вопрос, но он кажется немного другим, чем то, что я use - это услуга, а не действие, но, возможно, ответ аналогичен (связан с контекстом приложения)...

вот мой служебный код

public class BeaconsMonitoringService extends Service{

    private BeaconManager beaconManager;

    private String user;

    @Override
      public void onCreate() {
        // Configure BeaconManager.
        beaconManager = new BeaconManager(this);

      }

      @Override
      public int onStartCommand(Intent intent, int flags, int startId) {
          Toast.makeText(this, "Beacons monitoring service starting", Toast.LENGTH_SHORT).show();

          user = intent.getStringExtra("user");

            // Check if device supports Bluetooth Low Energy.
            if (!beaconManager.hasBluetooth()||!beaconManager.isBluetoothEnabled()) {
              Toast.makeText(this, "Device does not have Bluetooth Low Energy or it is not enabled", Toast.LENGTH_LONG).show();
              this.stopSelf();
            }

              connectToService();


          // If we get killed, after returning from here, restart
          return START_STICKY;
      }

      @Override
      public IBinder onBind(Intent intent) {
          // We don't provide binding, so return null
          return null;
      }

      @Override
      public void onDestroy() {
        Toast.makeText(this, "Beacons monitoring service done", Toast.LENGTH_SHORT).show();
      }

      private void connectToService() {


          beaconManager.connect(new BeaconManager.ServiceReadyCallback() {

            @Override
            public void onServiceReady() {
                notifyEnterRegion(0);
//            try {
                  beaconManager.setBackgroundScanPeriod(TimeUnit.SECONDS.toMillis(1), 0);
                  Log.i("BEACOON ", "ANTES DE");
                  beaconManager.setMonitoringListener(new MonitoringListener() {
                    @Override
                    public void onEnteredRegion(Region region, List<Beacon> beacons) {
                      Log.i("BEACOON ", String.valueOf(beacons.get(1).getMinor()));
                    for (Beacon beacon: beacons){
                        Log.i("BEACOON ", String.valueOf(beacon.getMinor()));
                        if (beacon.getMinor() == 64444) {

                            notifyEnterRegion(6444);

                        } else if (beacon.getMinor() == 36328) {

                            notifyEnterRegion(36328);

                        } else if (beacon.getMinor() == 31394) {

                            notifyEnterRegion(31394);

                        }
                    }
                    }

                    @Override
                    public void onExitedRegion(Region region) {

                        notifyExitRegion();

                    }
                  });  

            }
          });
        }

      public void notifyEnterRegion(int code) {

            Toast.makeText(this, "Beacon "+code, Toast.LENGTH_SHORT).show();

            Intent targetIntent = new Intent(this, MainActivity.class);
            PendingIntent contentIntent = PendingIntent.getActivity(this, 0, targetIntent, PendingIntent.FLAG_UPDATE_CURRENT);

            Notification noti = new Notification.Builder(this)
             .setContentTitle("Bienvenido "+user+"!")
             .setContentText("Sólo por estar aquí has ganado....")
             .setSmallIcon(com.smt.beaconssmt.R.drawable.beacon_gray)
             .setContentIntent(contentIntent)
             .getNotification();

            NotificationManager nManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
            nManager.notify(1, noti);
        }

      public void notifyExitRegion(){

            AlertDialog.Builder builder = new AlertDialog.Builder(this);

            builder.setMessage("Hasta pronto!")
                   .setTitle(user+", estás abandonando la zona de beacons");

            builder.setPositiveButton("Ver web", new DialogInterface.OnClickListener() {
                       public void onClick(DialogInterface dialog, int id) {
                           // User clicked OK button
                           Intent i = new Intent(BeaconsMonitoringService.this, WebViewActivity.class);
                           i.putExtra("web", "http://www.google.com/");
                           startActivity(i);
                       }
                   });
            builder.setNegativeButton("Adios!", new DialogInterface.OnClickListener() {
                       public void onClick(DialogInterface dialog, int id) {
                           // User cancelled the dialog
                       }
                   });

            AlertDialog dialog = builder.create();

            dialog.show();
      }



}

Буду очень признателен за любую помощь, заранее спасибо!


person Hugo    schedule 09.07.2014    source источник


Ответы (3)


Этот код работает для меня. Убедитесь, что вы указываете правильный UUID, младший и старший номера.

//Using something like that as global variable

    private static final Region[] BEACONS = new Region[] { 
    new Region("beacon1", "uuid1", 1, 19227), //uuid without "-"
    new Region("beacon2", "uuid2", 1, 61690),
    new Region("beacon3", "uuid3", null, null)
};
//Note: setting minor == null and major == null will detect every beacon with that uuid

  @Override
  public int onStartCommand(Intent intent, int flags, int startId) {
      startMonitoring();
      return START_STICKY;
  }



private void startMonitoring() {
    if (beaconManager == null) {
        beaconManager = new BeaconManager(this);

        // Configure verbose debug logging.
        L.enableDebugLogging(true);

        /**
         * Scanning
         */
        beaconManager.setBackgroundScanPeriod(TimeUnit.SECONDS.toMillis(1), 1);

        beaconManager.setRangingListener(new RangingListener() {

            @Override
            public void onBeaconsDiscovered(Region paramRegion, List<Beacon> paramList) {
                if (paramList != null && !paramList.isEmpty()) {
                    Beacon beacon = paramList.get(0);
                    Proximity proximity = Utils.computeProximity(beacon);
                    if (proximity == Proximity.IMMEDIATE) {
                        Log.d(TAG, "entered in region " + paramRegion.getProximityUUID());
                        postNotification(paramRegion);
                    } else if (proximity == Proximity.FAR) {
                        Log.d(TAG, "exiting in region " + paramRegion.getProximityUUID());
                         removeNotification(paramRegion);
                    }
                }
            }

        });

        beaconManager.connect(new BeaconManager.ServiceReadyCallback() {
            @Override
            public void onServiceReady() {
                try {
                    Log.d(TAG, "connected");
                    for (Region region : BEACONS) {
                        beaconManager.startRanging(region);
                    }
                } catch (RemoteException e) {
                    Log.d("TAG", "Error while starting monitoring");
                }
            }
        });
    }
}

*** Редактирование: код для вычисления точности

 public static double computeAccuracy(Beacon beacon)
 {
    if (beacon.getRssi() == 0) 
    {
       return -1.0D;
    }

    double ratio = beacon.getRssi() / beacon.getMeasuredPower();
    double rssiCorrection = 0.96D + Math.pow(Math.abs(beacon.getRssi()), 3.0D) % 10.0D / 150.0D;

    if (ratio <= 1.0D) 
    {
       return Math.pow(ratio, 9.98D) * rssiCorrection;
    }
    return (0.103D + 0.89978D * Math.pow(ratio, 7.71D)) * rssiCorrection;
 }

 public static Proximity proximityFromAccuracy(double accuracy)
 {
    if (accuracy < 0.0D) 
    {
        return Proximity.UNKNOWN;
    }
    if (accuracy < 0.5D) 
    {
        return Proximity.IMMEDIATE;
    }
    if (accuracy <= 3.0D) {
        return Proximity.NEAR;
    }
    return Proximity.FAR;
}



 public static Proximity computeProximity(Beacon beacon) {
     return proximityFromAccuracy(computeAccuracy(beacon));
 }
person jDur    schedule 09.07.2014
comment
Хорошо, это решение работает, спасибо! но у меня есть одно сомнение по этому поводу, как вы вычисляете близость? что-то в SDK или официальном коде или ваше собственное решение? Из этого я понимаю, что BeaconManager.setMonitoringListener() в этом случае не используется, и вместо этого мы должны использовать setRangingListener()?? - person Hugo; 09.07.2014
comment
Код для вычисления точности должен находиться в SDK. Но я использую пользовательскую реализацию SDK, поэтому я отредактировал ответ, чтобы показать вам код. О setMonitoringListener У меня не получилось заставить его работать, поэтому я заканчиваю использовать это решение. - person jDur; 09.07.2014
comment
Я интегрирую свое приложение Xamarin, здесь EnterRegion и ExitRegion вызывают несколько раз подряд, не знаю почему? Я читал об этом, он должен срабатывать только тогда, когда устройство попадает в зону действия региона. Но когда он входит в зону действия, он срабатывает последовательно, пока устройство не выйдет из зоны действия маяка. Пожалуйста, помогите разобраться с этим - person Ajay Sharma; 16.08.2017
comment
Вы не можете использовать Ranging в фоновом режиме постоянно. Это не очень хорошая практика. Это очень быстро разряжает батарею и ресурсы. Для фоновых целей используйте только мониторинг или, при необходимости, ранжирование в течение некоторого времени, а не постоянно или сочетание того и другого. Если вы все еще используете, позвольте мне сообщить, что ваше приложение может быть отклонено при публикации в магазине игр. - person Rishabh Chandel; 24.08.2017
comment
@AjaySharma, поскольку вы используете ранжирование, ваш триггер будет срабатывать для каждого периода сканирования, пока вы не окажетесь в зоне действия маяка. Это не очень хорошая практика. Использовать только мониторинг. - person Rishabh Chandel; 24.08.2017

Проверьте это, я надеюсь, что это поможет вам. Вы можете найти близость с помощью этой функции Utils.computeProximity(nearestBeacon).

beaconManager.setRangingListener(new BeaconManager.RangingListener() {

        @Override
        public void onBeaconsDiscovered(Region region, final List<Beacon> beaconList) {
            if (!beaconList.isEmpty()) {
                Beacon nearestBeacon = beaconList.get(0);
                Log.e("Current Proximity - ", String.valueOf(Utils.computeProximity(nearestBeacon)));
            }
        }
});
person Android Develeoper    schedule 28.12.2016

Из вашего сервиса вы также можете использовать эту библиотеку для поиска iBeacons. Это может сэкономить вашим пользователям батарею. Я также предпочел бы это, поскольку BeaconManager фактически порождает свою собственную службу, и поэтому у вас есть 2 службы, работающие только для сканирования iBeacons. Однако в нем отсутствует вычисление расстояния.

Также имейте в виду, что для сканирования iBeacons вам необходимо:

  • Наличие чипа Bluetooth LE: любой.
  • Включить Bluetooth: любой.
  • Иметь местоположение на: Android 6+.
  • Иметь разрешение на выполнение местоположения: Android 6+.
  • Максимальный запуск 5 сканирований за 30 секунд: Android 7+.

И чтобы ваша служба работала, вам нужно будет добавить следующие слушатели вещания:

  • состояние локации изменено
  • состояние блютуза изменилось
  • загрузочный приемник

С этих приемников вы должны снова начать свою службу.

person Elias    schedule 29.12.2016