当前位置: 移动技术网 > IT编程>移动开发>Android > Android 百度地图定位实现仿钉钉签到打卡功能的完整代码

Android 百度地图定位实现仿钉钉签到打卡功能的完整代码

2020年06月23日  | 移动技术网IT编程  | 我要评论

北京展馆,文秘家园,杰拉德格林面补

导语

本章根据百度地图api,实现仿钉钉打卡功能。用到了基础地图、覆盖物、定位图层、陀螺仪方法、悬浮信息弹框。

百度地图api地址  :

请先注册注册百度账号和获取密钥,并实现地图显示出来。(注意:密钥、权限要设置)

另外,我得说明本章所下载官方demo 和 导入的jar包和so文件。自定义下载即可,如下图:

接下来,一起看实现效果。

源码git地址:baidumapapp

效果图        

   

实现代码·三步骤

第一步:基础地图和方向传感器

类先实现方向传感器 implements sensoreventlistener

@override
 public void onsensorchanged(sensorevent sensorevent) {
  double x = sensorevent.values[sensormanager.data_x];
  if (math.abs(x - lastx) > 1.0) {
   mcurrentdirection = (int) x;
   locdata = new mylocationdata.builder()
     // 此处设置开发者获取到的方向信息,顺时针0-360
     .direction(mcurrentdirection).latitude(mcurrentlat)
     .longitude(mcurrentlon).build();
   mbaidumap.setmylocationdata(locdata);
  }
  lastx = x;
 }
 
 @override
 public void onaccuracychanged(sensor sensor, int i) {
 
 }
 /**
  * 初始化地图
  */
 private void initbaidumap() {
  mmapview = (mapview) findviewbyid(r.id.mapview);
  mbaidumap = mmapview.getmap();
  mbaidumap.setmaptype(baidumap.map_type_normal);
  mbaidumap.setmylocationenabled(true);//开启定位图层
  msensormanager = (sensormanager) getsystemservice(sensor_service);//获取传感器管理服务
 }
@override
 protected void onresume() {
  super.onresume();
  mmapview.onresume();
  //为系统的方向传感器注册监听器
  msensormanager.registerlistener(this, msensormanager.getdefaultsensor(sensor.type_orientation),
    sensormanager.sensor_delay_ui);
 }
 @override
 protected void onpause() {
  super.onpause();
  mmapview.onpause();
 }
 
 @override
 protected void onstop() {
  super.onstop();
  //取消注册传感器监听
  msensormanager.unregisterlistener(this);
 }

第二步:开启定位

 /***
  * 定位选项设置
  * @return
  */
 public void getlocationclientoption() {
  moption = new locationclientoption();
  moption.setlocationmode(locationclientoption.locationmode.hight_accuracy);//可选,默认高精度,设置定位模式,高精度,低功耗,仅设备
  moption.setcoortype("bd09ll");//可选,默认gcj02,设置返回的定位结果坐标系,如果配合百度地图使用,建议设置为bd09ll;
  moption.setscanspan(2000);//可选,默认0,即仅定位一次,设置发起连续定位请求的间隔需要大于等于1000ms才是有效的
  moption.setisneedaddress(true);//可选,设置是否需要地址信息,默认不需要
  moption.setisneedlocationdescribe(true);//可选,设置是否需要地址描述
  moption.setneeddevicedirect(true);//可选,设置是否需要设备方向结果
  moption.setlocationnotify(true);//可选,默认false,设置是否当gps有效时按照1s1次频率输出gps结果
  moption.setignorekillprocess(true);//可选,默认true,定位sdk内部是一个service,并放到了独立进程,设置是否在stop的时候杀死这个进程,默认不杀死
  moption.setisneedlocationdescribe(false);//可选,默认false,设置是否需要位置语义化结果,可以在bdlocation.getlocationdescribe里得到,结果类似于“在北京天安门附近”
  moption.setisneedlocationpoilist(false);//可选,默认false,设置是否需要poi结果,可以在bdlocation.getpoilist里得到
  moption.setignorecacheexception(false);//可选,默认false,设置是否收集crash信息,默认收集
  moption.setopengps(true);//可选,默认false,设置是否开启gps定位
  moption.setisneedaltitude(false);//可选,默认false,设置定位时是否需要海拔信息,默认不需要,除基础定位版本都可用
  client = new locationclient(this);
  client.setlocoption(moption);
  client.registerlocationlistener(bdablistener);
  client.start();
 }
 /***
  * 接收定位结果消息,并显示在地图上
  */
 private bdabstractlocationlistener bdablistener = new bdabstractlocationlistener() {
  @override
  public void onreceivelocation(bdlocation location) {
   //定位方向
   mcurrentlat = location.getlatitude();
   mcurrentlon = location.getlongitude();
   //个人定位
   locdata = new mylocationdata.builder()
     .direction(mcurrentdirection).latitude(location.getlatitude())
     .longitude(location.getlongitude()).build();
   mbaidumap.setmylocationdata(locdata);
   mbaidumap.setmylocationconfiguration(new mylocationconfiguration(
     mylocationconfiguration.locationmode.normal, true, null));
   //更改ui
   message message = new message();
   message.obj = location;
   mhandler.sendmessage(message);
  }
 };

第三步:更改ui

 //设置打卡目标范围圈
 private void setcircleoptions() {
  if (mdestinationpoint == null) return;//打卡坐标不能为空
  overlayoptions oocircle = new circleoptions().fillcolor(0x4057fff8)
    .center(mdestinationpoint).stroke(new stroke(1, 0xb6ffffff)).radius(distance);
  mbaidumap.addoverlay(oocircle);
 }
 /**
  * 添加地图文字
  *
  * @param point
  * @param str
  * @param color 字体颜色
  */
 private void settextoption(latlng point, string str, string color) {
  //使用makerinfowindow
  if (point == null) return;
  textview view = new textview(getapplicationcontext());
  view.setbackgroundresource(r.mipmap.map_textbg);
  view.setpadding(0, 23, 0, 0);
  view.settypeface(typeface.default_bold);
  view.settextsize(14);
  view.setgravity(gravity.center);
  view.settext(str);
  view.settextcolor(color.parsecolor(color));
  minfowindow = new infowindow(view, point, 170);
  mbaidumap.showinfowindow(minfowindow);
 }
 /**
  * 设置marker覆盖物
  *
  * @param ll 坐标
  * @param icon 图标
  */
 private void setmarkeroptions(latlng ll, int icon) {
  if (ll == null) return;
  bitmapdescriptor bitmap = bitmapdescriptorfactory.fromresource(icon);
  markeroptions ood = new markeroptions().position(ll).icon(bitmap);
  mbaidumap.addoverlay(ood);
 }
 //改变地图缩放
 private void setmapzoomscale(latlng ll) {
  if (mdestinationpoint == null) {//打卡坐标不为空
   mzoomscale = getzoomscale(ll);
   mbaidumap.animatemapstatus(mapstatusupdatefactory.newlatlngzoom(ll, mzoomscale));//缩放
  } else {
   mzoomscale = getzoomscale(ll);
   mbaidumap.animatemapstatus(mapstatusupdatefactory.newlatlngzoom(mcenterpos, mzoomscale));//缩放
  }
 }
 /**
  * 获取地图的中心点和缩放比例
  *
  * @return float
  */
 private float getzoomscale(latlng locationpoint) {
  double maxlong; //最大经度
  double minlong; //最小经度
  double maxlat;  //最大纬度
  double minlat;  //最小纬度
  list<double> longitems = new arraylist<double>(); //经度集合
  list<double> latitems = new arraylist<double>();  //纬度集合
 
  if (null != locationpoint) {
   longitems.add(locationpoint.longitude);
   latitems.add(locationpoint.latitude);
  }
  if (null != mdestinationpoint) {
   longitems.add(mdestinationpoint.longitude);
   latitems.add(mdestinationpoint.latitude);
  }
 
  maxlong = longitems.get(0); //最大经度
  minlong = longitems.get(0); //最小经度
  maxlat = latitems.get(0);  //最大纬度
  minlat = latitems.get(0);  //最小纬度
 
  for (int i = 0; i < longitems.size(); i++) {
   maxlong = math.max(maxlong, longitems.get(i)); //获取集合中的最大经度
   minlong = math.min(minlong, longitems.get(i)); //获取集合中的最小经度
  }
 
  for (int i = 0; i < latitems.size(); i++) {
   maxlat = math.max(maxlat, latitems.get(i)); //获取集合中的最大纬度
   minlat = math.min(minlat, latitems.get(i)); //获取集合中的最小纬度
  }
  double latcenter = (maxlat + minlat) / 2;
  double longcenter = (maxlong + minlong) / 2;
  int jl = (int) getdistance(new latlng(maxlat, maxlong), new latlng(minlat, minlong));//缩放比例参数
  mcenterpos = new latlng(latcenter, longcenter); //获取中心点经纬度
  int zoomlevel[] = {2500000, 2000000, 1000000, 500000, 200000, 100000,
    50000, 25000, 20000, 10000, 5000, 2000, 1000, 500, 100, 50, 20, 0};
  int i;
  for (i = 0; i < 18; i++) {
   if (zoomlevel[i] < jl) {
    break;
   }
  }
  float zoom = i + 4;
  return zoom;
 }
 
 /**
  * 缩放比例参数
  *
  * @param var0
  * @param var1
  * @return
  */
 public double getdistance(latlng var0, latlng var1) {
  if (var0 != null && var1 != null) {
   point var2 = coordutil.ll2point(var0);
   point var3 = coordutil.ll2point(var1);
   return var2 != null && var3 != null ? coordutil.getdistance(var2, var3) : -1.0d;
  } else {
   return -1.0d;
  }
 }
 /**
  * 处理连续定位的地图ui变化
  */
 private handler mhandler = new handler() {
  @override
  public void handlemessage(message msg) {
   super.handlemessage(msg);
   bdlocation location = (bdlocation) msg.obj;
   latlng locationpoint = new latlng(location.getlatitude(), location.getlongitude());
   //打卡范围
   mdestinationpoint = new latlng(location.getlatitude() * 1.0001, location.getlongitude() * 1.0001);//假设公司坐标
   setcircleoptions();
   //计算两点距离,单位:米
   mdistance = distanceutil.getdistance(mdestinationpoint, locationpoint);
   if (mdistance <= distance) {
    //显示文字
    settextoption(mdestinationpoint, "您已在餐厅范围内", "#7ed321");
    //目的地图标
    setmarkeroptions(mdestinationpoint, r.mipmap.arrive_icon);
    //按钮颜色
    //commit_bt.setbackgrounddrawable(getresources().getdrawable(r.mipmap.restaurant_btbg_yellow));
    mbaidumap.setmylocationenabled(false);
   } else {
    settextoption(locationpoint, "您不在餐厅范围之内", "#ff6c6c");
    setmarkeroptions(mdestinationpoint, r.mipmap.restaurant_icon);
    //commit_bt.setbackgrounddrawable(getresources().getdrawable(r.mipmap.restaurant_btbg_gray));
    mbaidumap.setmylocationenabled(true);
   }
   // mdistance_tv.settext("距离目的地:" + mdistance + "米");
   //缩放地图
   setmapzoomscale(locationpoint);
  }
 };

实现时间显示

 /**
  * 设置系统时间
  */
 private runnable run = new runnable() {
  @override
  public void run() {
   simpledateformat simpledateformat = new simpledateformat("hh:mm:ss");// hh:mm:ss
   date date = new date(system.currenttimemillis());//获取当前时间
   mtime_tv.settext(simpledateformat.format(date)); //更新时间
   mhandler.postdelayed(run, 1000);
  }
 };
  mhandler.post(run);//设置系统时间

最后,收尾操作

 @override
 protected void ondestroy() {
  if (bdablistener != null) {
   client.unregisterlocationlistener(bdablistener);
 
  }
  if (client != null && client.isstarted()) {
   client.stop();
  }
  mmapview.ondestroy();
  mmapview = null;
  mhandler.removecallbacks(run);
  super.ondestroy();
 }

源码地址:https://github.com/aiyangtianci/baidumapapp

总结

到此这篇关于android 百度地图定位实现仿钉钉签到打卡功能的文章就介绍到这了,更多相关android 钉钉签到打卡内容请搜索移动技术网以前的文章或继续浏览下面的相关文章希望大家以后多多支持移动技术网!

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网