현재 서비스 중인 앱에서 wifi를 활용하여 출결관리를 하고있다.
wifi를 잡았을때만 출결관리가 되기때문에 와이파이가 도달하지 않는 구역에서는 앱으로 출결관리를 하기 어렵다.
하지만 비콘(beacon)을 활용하면 비콘을 회사내의 여러군대(출입구,탈의실,화잘실..?) 에 설치해 두면
어느곳에서든 출결체크가 가능할것같아 테스트한다.
먼저 테스트 환경은 아래와같다.
Tool : Android Studio
Gradle Version : 5.4.1
Min sdk : 18
Targetsdk : 29
beacon library : alt-beacon(https://altbeacon.github.io/android-beacon-library/)
비콘은 최저가 아무거나 사면된다...
(단.. 너무 싼거는 의심해보아야함. 처음 너무싼거 (1+1)샀다가 비콘 내부 세팅 못함. 제조업체 망해서 더이상 제공불가)
대부분의 비콘은
Beacon SET(https://play.google.com/store/apps/details?id=com.minnw.beaconset&hl=ko)
의 어플로 셋팅가능하나 간혹 제조사에서 막아놓은경우 제조사에서 제공하는 어플로 셋팅해야 됨.(아이폰 동일)
alt-beacon 라이브러리 사용 하여 테스트
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation 'androidx.appcompat:appcompat:1.0.2'
implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'androidx.test.ext:junit:1.1.0'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
implementation 'org.altbeacon:android-beacon-library:2.+'//alt-beacon library
}
build.gradle 에서 dependencies 내부 'org.altbeacon:android-beacon-librarary:2.+' 삽입
테스트 버전이므로 MainActivity 에 모든 기능 삽입(추후 필요에따라 모듈화하여 변경예정)
package com.example.beacontest;
import androidx.appcompat.app.AlertDialog;
import androidx.appcompat.app.AppCompatActivity;
import android.Manifest;
import android.content.DialogInterface;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.os.RemoteException;
import android.util.Log;
import android.view.View;
import android.widget.TextView;
import org.altbeacon.beacon.Beacon;
import org.altbeacon.beacon.BeaconConsumer;
import org.altbeacon.beacon.BeaconManager;
import org.altbeacon.beacon.BeaconParser;
import org.altbeacon.beacon.RangeNotifier;
import org.altbeacon.beacon.Region;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
public class MainActivity extends AppCompatActivity implements BeaconConsumer {
private static final String TAG = "Beacontest";
private BeaconManager beaconManager;
private List<Beacon> beaconList = new ArrayList<>();
TextView textView;
private static final int PERMISSION_REQUEST_COARSE_LOCATION = 1;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//비콘 매니저 생성,
beaconManager = BeaconManager.getInstanceForApplication(this);
textView = (TextView) findViewById(R.id.Textview);//비콘검색후 검색내용 뿌려주기위한 textview
//비콘 매니저에서 layout 설정 'm:2-3=0215,i:4-19,i:20-21,i:22-23,p:24-24,d:25-25'
beaconManager.getBeaconParsers().add(new BeaconParser().setBeaconLayout("m:2-3=0215,i:4-19,i:20-21,i:22-23,p:24-24,d:25-25"));
//beaconManager 설정 bind
beaconManager.bind(this);
//beacon 을 활용하려면 블루투스 권한획득(Andoird M버전 이상)
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.M){
if(this.checkSelfPermission(Manifest.permission.ACCESS_COARSE_LOCATION)!=PackageManager.PERMISSION_GRANTED){
final AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("This app needs location access" );
builder.setMessage("Please grant location access so this app can detect beacons.");
builder.setPositiveButton(android.R.string.ok,null);
builder.setOnDismissListener(new DialogInterface.OnDismissListener() {
@Override
public void onDismiss(DialogInterface dialogInterface) {
requestPermissions(new String[]{Manifest.permission.ACCESS_COARSE_LOCATION},PERMISSION_REQUEST_COARSE_LOCATION);
}
});
builder.show();
}
}
}
@Override
protected void onDestroy() {
super.onDestroy();
beaconManager.unbind(this);
}
@Override
public void onBeaconServiceConnect() {
beaconManager.addRangeNotifier(new RangeNotifier() {
@Override
// 비콘이 감지되면 해당 함수가 호출된다. Collection<Beacon> beacons에는 감지된 비콘의 리스트가,
// region에는 비콘들에 대응하는 Region 객체가 들어온다.
public void didRangeBeaconsInRegion(Collection<Beacon> beacons, Region region) {
if (beacons.size() > 0) {
beaconList.clear();
for (Beacon beacon : beacons) {
beaconList.add(beacon);
}
}
}
});
try {
beaconManager.startRangingBeaconsInRegion(new Region("myRangingUniqueId", null, null, null));
} catch (RemoteException e) { }
}
// 버튼이 클릭되면 textView 에 비콘들의 정보를 뿌린다.
public void OnButtonClicked(View view){
// 아래에 있는 handleMessage를 부르는 함수. 맨 처음에는 0초간격이지만 한번 호출되고 나면
// 1초마다 불러온다.
handler.sendEmptyMessage(0);
}
Handler handler = new Handler() {
public void handleMessage(Message msg) {
textView.setText("");
// 비콘의 아이디와 거리를 측정하여 textView에 넣는다.
for(Beacon beacon : beaconList){
String uuid=beacon.getId1().toString(); //beacon uuid
int major = beacon.getId2().toInt(); //beacon major
int minor = beacon.getId3().toInt();// beacon minor
String address = beacon.getBluetoothAddress();
if(major==40001){
//beacon 의 식별을 위하여 major값으로 확인
//이곳에 필요한 기능 구현
//textView.append("ID 1 : " + beacon.getId2() + " / " + "Distance : " + Double.parseDouble(String.format("%.3f", beacon.getDistance())) + "m\n");
textView.append("출근하셔야되는데...\n");
textView.append("Beacon Bluetooth Id : "+address+"\n");
textView.append("Beacon UUID : "+uuid+"\n");
}else{
//나머지 비콘검색
textView.append("ID 2: " + beacon.getId2() + " / " + "Distance : " + Double.parseDouble(String.format("%.3f", beacon.getDistance())) + "m\n");
}
}
// 자기 자신을 1초마다 호출
handler.sendEmptyMessageDelayed(0, 1000);
}
};
@Override
public void onRequestPermissionsResult(int requestCode,
String permissions[], int[] grantResults) {
switch (requestCode) {
case PERMISSION_REQUEST_COARSE_LOCATION: {
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
Log.d(TAG, "coarse location permission granted");
} else {
final AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("Functionality limited");
builder.setMessage("Since location access has not been granted, this app will not be able to discover beacons when in the background.");
builder.setPositiveButton(android.R.string.ok, null);
builder.setOnDismissListener(new DialogInterface.OnDismissListener() {
@Override
public void onDismiss(DialogInterface dialog) {
}
});
builder.show();
}
return;
}
}
}
}
해당소스 휴대폰에 빌드후 실행하면 아래와같은 결과값을 얻을수있다.
비콘 테스트는 꼭 휴대폰에서 해야함.
에뮬에서는 비콘 테스트가 되지않음(블루투스 때문인것 같음)
'코딩쟁이 > android' 카테고리의 다른 글
[안드로이드] 오레오 대응 Notification Channel, FCM삽질기 (0) | 2019.11.07 |
---|---|
[안드로이드]FCM 푸시울릴때 화면깨우기. (0) | 2019.10.08 |
[WebView]shouldOverrideUrlLoading (0) | 2017.12.12 |