I use the AltBEacon Android library to develop the iBeacon application for Android devices . I look through the lighthouses, however, only two of the four lighthouses are found (sometimes 1/4).
I increase mBeaconManager.setForegroundScanPeriod(5000l);to 5 seconds, but still the same result. I'm not sure if using CustomAdapterto bind data to view is wrong or a problem with a mobile device (I use Galaxy Note II - Android 4.4.2 (KitKat))? Can anyone find my mistake?
Another problem is that the calculation of the distance from the device to the beacon returns incorrectly (when I am approximately 0.5 m, it returns betweem 3-6 m)
What am I doing wrong?
Note. I checked the UUID, major, minor beacon for any mistake. I increased the scan time. None of them worked.
Here is a snippet that starts monitoring and ranking:
public class FragmentScanBeacons extends Fragment implements BeaconConsumer{
public static Region mRegion = new Region("Server", Identifier.parse("Here is my UUID"), null, null);
private BeaconManager mBeaconManager;
private BeaconBaseAdapter beaconBaseAdapter;
private ListView beaconsListLv;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mBeaconManager = BeaconManager.getInstanceForApplication(getActivity());
mBeaconManager.getBeaconParsers().add(new BeaconParser().setBeaconLayout("m:2-3=0215,i:4-19,i:20-21,i:22-23,p:24-24"));
beaconBaseAdapter = new BeaconBaseAdapter(getActivity());
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_beacons, container, false);
beaconsListLv = (ListView) view.findViewById(R.id.beaconsListView);
beaconsListLv.setAdapter(beaconBaseAdapter);
verifyBluetooth();
mBeaconManager.bind(this);
return view;
}
@Override
public void onResume() {
super.onResume();
if(mBeaconManager.isBound(this)){
mBeaconManager.setBackgroundMode(false);
}
}
@Override
public void onPause() {
super.onPause();
if(mBeaconManager.isBound(this)){
mBeaconManager.setBackgroundMode(true);
}
}
@Override
public void onDestroy() {
super.onDestroy();
mBeaconManager.unbind(this);
}
@Override
public void onBeaconServiceConnect() {
try {
mBeaconManager.setForegroundScanPeriod(1000l);
mBeaconManager.setForegroundBetweenScanPeriod(0l);
mBeaconManager.updateScanPeriods();
}
catch (RemoteException e){
e.printStackTrace();
}
mBeaconManager.setMonitorNotifier(new MonitorNotifier() {
@Override
public void didEnterRegion(Region region) {
Log.d("TEST", "ENTERED beacon region");
try {
mBeaconManager.startRangingBeaconsInRegion(mRegion);
}
catch (RemoteException e) {
e.printStackTrace();
}
}
@Override
public void didExitRegion(Region region) {
Log.d("TEST", "EXITED beacon region");
}
@Override
public void didDetermineStateForRegion(int state, Region region) {
Log.d("TEST", "SWITCHED from seeing/not seeing beacon to state " + state);
}
});
mBeaconManager.setRangeNotifier(new RangeNotifier() {
@Override
public void didRangeBeaconsInRegion(final Collection<Beacon> beacons, Region region) {
if (beacons != null && beacons.size() > 0) {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
beaconBaseAdapter.initAll(beacons);
}
});
}
}
});
try {
mBeaconManager.startMonitoringBeaconsInRegion(mRegion);
}
catch (RemoteException e) {
e.printStackTrace();
}
}
@Override
public Context getApplicationContext() {
return getActivity().getApplicationContext();
}
@Override
public void unbindService(ServiceConnection serviceConnection) {
getActivity().unbindService(serviceConnection);
}
@Override
public boolean bindService(Intent intent, ServiceConnection serviceConnection, int mode) {
return getActivity().bindService(intent, serviceConnection, mode);
}
}
And here is the custom adapter:
public class BeaconBaseAdapter extends BaseAdapter {
private Context myContext;
private LayoutInflater inflater;
public static ArrayList<Beacon> beacons;
public BeaconBaseAdapter(Context context) {
this.myContext = context;
this.inflater = LayoutInflater.from(context);
this.beacons = new ArrayList<Beacon>();
}
public void initAll(Collection<Beacon> newBeacons) {
this.beacons.clear();
this.beacons.addAll(newBeacons);
notifyDataSetChanged();
}
@Override
public int getCount() {
return beacons.size();
}
@Override
public Beacon getItem(int position) {
return beacons.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = inflater.inflate(R.layout.beacon_list_row, null);
convertView.setTag(new ViewHolder(convertView));
}
bind(getItem(position), position, convertView);
return convertView;
}
private void bind(Beacon beacon, int position, View view) {
ViewHolder holder = (ViewHolder) view.getTag();
holder.manufacturerTextView.setText("Manufacturer: " + beacon.getManufacturer());
holder.idOneTextView.setText("UUID: " + beacon.getId1());
holder.idTwoTextView.setText("Major: " + beacon.getId2());
holder.idThreeTextView.setText("Minor: " + beacon.getId3());
holder.txPowerTextView.setText("TX-Power: " + beacon.getTxPower());
holder.rssiTextView.setText("RSSI: " + beacon.getRssi());
holder.distanceTextView.setText(String.format("DISTANCE: (%.2f m)", beacon.getDistance()));
holder.nameTextView.setText("Bluetooth Name: " + beacon.getBluetoothName());
holder.addressTextView.setText("Bluetooth Adrs: " + beacon.getBluetoothAddress());
}
static class ViewHolder {
final TextView nameTextView;
final TextView manufacturerTextView;
final TextView idOneTextView;
final TextView idTwoTextView;
final TextView idThreeTextView;
final TextView txPowerTextView;
final TextView rssiTextView;
final TextView distanceTextView;
final TextView addressTextView;
ViewHolder(View view) {
nameTextView = (TextView) view.findViewWithTag("name");
manufacturerTextView = (TextView) view.findViewWithTag("manufacturer");
idOneTextView = (TextView) view.findViewWithTag("id_one");
idTwoTextView = (TextView) view.findViewWithTag("id_two");
idThreeTextView = (TextView) view.findViewWithTag("id_three");
txPowerTextView = (TextView) view.findViewWithTag("tx_power");
rssiTextView = (TextView) view.findViewWithTag("rssi");
distanceTextView = (TextView) view.findViewWithTag("distance");
addressTextView = (TextView) view.findViewWithTag("address");
}
}
}