okhttp3+xrecyclerview+详情页面+高德地图定位

本文介绍了一个基于Android的应用如何集成高德地图API实现位置定位功能,包括依赖配置、权限设置、布局文件设计及代码实现等关键步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、依赖

compile 'com.google.code.gson:gson:2.6.2'
compile 'com.squareup.okhttp3:okhttp:3.3.0'
compile 'com.github.bumptech.glide:glide:3.7.0'
compile 'com.jcodecraeer:xrecyclerview:1.5.9'
implementation files('libs/AMap_Location_V3.8.0_20180201.jar')
implementation files('libs/Amap_2DMap_V5.2.0_20170627.jar')
implementation files('libs/Android_Map3D_SDK_V6.0.1_20180309.jar')

二、权限

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.test">

    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <!--用于进行网络定位-->
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission>
    <!--用于访问GPS定位-->
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>
    <!--用于获取运营商信息,用于支持提供运营商信息相关的接口-->
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>
    <!--用于访问wifi网络信息,wifi信息会用于进行网络定位-->
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>
    <!--用于获取wifi的获取权限,wifi信息会用来进行网络定位-->
    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE"></uses-permission>
    <!--用于访问网络,网络定位需要上网-->
    <uses-permission android:name="android.permission.INTERNET"></uses-permission>
    <!--用于读取手机当前的状态-->
    <uses-permission android:name="android.permission.READ_PHONE_STATE"></uses-permission>
    <!--用于写入缓存数据到扩展存储卡-->
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
    <!--用于申请调用A-GPS模块-->
    <uses-permission android:name="android.permission.ACCESS_LOCATION_EXTRA_COMMANDS"></uses-permission>
    <!--用于申请获取蓝牙信息进行室内定位-->
    <uses-permission android:name="android.permission.BLUETOOTH"></uses-permission>
    <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"></uses-permission>

    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <!--允许程序访问WiFi网络信息-->
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <!--允许程序读写手机状态和身份-->
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <!--允许程序访问CellID或WiFi热点来获取粗略的位置-->
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".view.MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name=".view.DetailActivity"></activity>
        <activity android:name=".view.MyAddressActivity"></activity>
        <service android:name="com.amap.api.location.APSService"></service>

        <meta-data android:name="com.amap.api.v2.apikey" android:value="a8e7d0111e7e9f434ae1aab2b0dc6cc7">
        </meta-data>
    </application>

</manifest>

三、布局

activity_main.xml

<com.jcodecraeer.xrecyclerview.XRecyclerView
        android:id="@+id/recylerview"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
</com.jcodecraeer.xrecyclerview.XRecyclerView>

activity_detail.xml

<ImageView

    android:id="@+id/detail_pic"
    android:layout_width="match_parent"
    android:layout_height="300dp"
    android:scaleType="centerCrop" />

<TextView
    android:id="@+id/detail_title"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:text="商品名" />

<TextView
    android:id="@+id/detail_address"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:text="北京" />

address_layout.xml(地图页面布局)

<com.amap.api.maps2d.MapView
    android:id="@+id/map"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

layout_item.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="horizontal">
    <ImageView
        android:id="@+id/item_pic"
        android:layout_width="60dp"
        android:layout_height="60dp"
        android:src="@mipmap/ic_launcher" />
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical">
        <TextView
            android:id="@+id/item_title"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="商品名称"
            android:textSize="25sp" />
        <TextView
            android:id="@+id/item_price"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="商品价格"
            android:textSize="20sp" />
    </LinearLayout>
</LinearLayout>

四、代码(四个包:http、model、presenter、view)


MainActivity

  import android.content.Intent;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.LinearLayoutManager;
import android.util.Log;
import android.view.View;
import android.view.Window;
import com.example.shixun1_monthtest.R;
import com.example.shixun1_monthtest.model.GoodsBean;
import com.example.shixun1_monthtest.model.ModelImpl;
import com.example.shixun1_monthtest.model.MyAdapter;
import com.example.shixun1_monthtest.presenter.PresenterImpl;
import com.jcodecraeer.xrecyclerview.XRecyclerView;
import java.util.List;

public class MainActivity extends AppCompatActivity implements IMainView {

    private XRecyclerView recyclerView;
    private static final String TAG = "MainActivity----";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.activity_main);
        initViews();
        initData();
    }
    private void initData() {
        PresenterImpl presenter = new PresenterImpl();
        presenter.showGoodsListToView(new ModelImpl(), this);
    }

    private void initViews() {
        recyclerView = findViewById(R.id.recylerview);
        //设置
        recyclerView.setLayoutManager(new LinearLayoutManager(this));
    }

    //显示数据的方法
    @Override
    public void showGoodsList(final List<GoodsBean.DataBean> list) {
        Log.d(TAG, "showGoodsList: " + list);
        MyAdapter myAdapter = new MyAdapter(this, list);
        myAdapter.setMyItemClickListener(new MyAdapter.MyItemClickListener() {
            @Override
            public void onItemClick(View view, int postion) {
                Intent intent = new Intent(MainActivity.this, DetailActivity.class);
                String pic_url = list.get(postion).getImages().split("\\|")[0];
                intent.putExtra("pic_url", pic_url);
                intent.putExtra("title", list.get(postion).getTitle());
                startActivity(intent);
                finish();
            }
        });
        recyclerView.setAdapter(myAdapter);
    }
}

model下


GoodsBean(用接口请求数据)
(   https://www.zhaoapi.cn/product/getProducts?pscid=1   )


IModel

public interface IModel {
    //获取商品数据
    void getGoodsList(String url, GetGoodsListener getGoodsListener);
}

MyAdapter

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
import com.bumptech.glide.Glide;
import com.example.shixun1_monthtest2.R;
import java.util.List;
/**
 * Created by gjl on 2018/4/2.
 */
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> {

    private final Context context;
    private final List<GoodsBean.DataBean> list;
    public MyItemClickListener myItemClickListener;

    public MyAdapter(Context context, List<GoodsBean.DataBean> list) {
        this.context = context;
        this.list = list;
    }
    @Override
    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = View.inflate(context, R.layout.layout_item, null);
        MyViewHolder myViewHolder = new MyViewHolder(view);
        return myViewHolder;
    }
    @Override
    public void onBindViewHolder(MyViewHolder holder, final int position) {
        String pic_url = list.get(position).getImages().split("\\|")[0];
        Glide.with(context).load(pic_url).into(holder.getItem_pic());
        holder.getItem_title().setText(list.get(position).getTitle());
        holder.getItem_price().setText("价格:"+list.get(position).getBargainPrice());
        //这是条目点击事件
        holder.itemView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                myItemClickListener.onItemClick(v,position);
            }
        });
    }
    @Override
    public int getItemCount() {
        return list.size();
    }
    //ViewHodler
    class MyViewHolder extends RecyclerView.ViewHolder {
        private final ImageView item_pic;
        private final TextView item_title,item_price;
        public MyViewHolder(View itemView) {
            super(itemView);
            item_pic = itemView.findViewById(R.id.item_pic);
            item_title = itemView.findViewById(R.id.item_title);
            item_price = itemView.findViewById(R.id.item_price);
        }
        public ImageView getItem_pic() {
            return item_pic;
        }
        public TextView getItem_title() {
            return item_title;
        }
        public TextView getItem_price() {
            return item_price;
        }
    }
    //自己定义条目点击事件
    public interface MyItemClickListener{
        void onItemClick(View view, int postion);
    }
    public void setMyItemClickListener(MyItemClickListener myItemClickListener){
        this.myItemClickListener = myItemClickListener;
    }
}

View下 IMainView

import com.example.shixun1_monthtest.model.GoodsBean;

import java.util.List;
public interface IMainView {
    //显示商品列表的方法
    void showGoodsList(List<GoodsBean.DataBean> list);

}

presenter包下 IPresenter

import com.example.shixun1_monthtest.model.IModel;
import com.example.shixun1_monthtest.view.IMainView;
public interface IPresenter {
    void showGoodsListToView(IModel iModel, IMainView iMainView);
}

PresenterImpl

import com.example.shixun1_monthtest.http.HttpConfig;
import com.example.shixun1_monthtest.model.GetGoodsListener;
import com.example.shixun1_monthtest.model.GoodsBean;
import com.example.shixun1_monthtest.model.IModel;
import com.example.shixun1_monthtest.view.IMainView;
import com.google.gson.Gson;
import java.util.List;
public class PresenterImpl implements IPresenter {
    @Override
    public void showGoodsListToView(IModel iModel, final IMainView iMainView) {

        String url = HttpConfig.url+"?pscid=1";

        iModel.getGoodsList(url, new GetGoodsListener() {
            @Override
            public void getGoodsSuccess(String json) {
                Gson gson=new Gson();
                GoodsBean goodsBean = gson.fromJson(json, GoodsBean.class);
                List<GoodsBean.DataBean> data = goodsBean.getData();
                //将数据传入View
                iMainView.showGoodsList(data);
            }
            @Override
            public void getGoodsError(String error) {
            }
        });
    }
}

http包下

HttpConfig

public class HttpConfig {
    public static String url = "https://www.zhaoapi.cn/product/getProducts";
}

OkHttpUitls

import android.os.Handler;
import android.os.Message;
import java.io.IOException;
import okhttp3.Call;
import okhttp3.Callback;
import okhttp3.FormBody;
import okhttp3.Interceptor;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.RequestBody;
import okhttp3.Response;
public class OkHttpUitls {
    private static OkHttpUitls okHttpUitls;
    private MyHandler myHandler = new MyHandler();
    private OkLoadListener okLoadListener;
    public static OkHttpUitls getOkHttpUitls() {
        if (okHttpUitls == null) {
            okHttpUitls = new OkHttpUitls();
        }
        return okHttpUitls;
    }
    //get
    public void okGet(String url) {
        OkHttpClient okHttpClient = new OkHttpClient.Builder().addInterceptor(new MyIntercepter()).build();
        Request request = new Request.Builder().url(url).build();
        Call call = okHttpClient.newCall(request);
        call.enqueue(new Callback() {
            @Override
            public void onFailure(Call call, IOException e) {
                Message message = myHandler.obtainMessage();
                message.what = 0;
                message.obj = e.getMessage();
                myHandler.sendMessage(message);
            }
            @Override
            public void onResponse(Call call, Response response) throws IOException {
                Message message = myHandler.obtainMessage();
                message.what = 1;
                message.obj = response.body().string();
                myHandler.sendMessage(message);
            }
        });
    }
    //post
    //拦截器
    class MyIntercepter implements Interceptor {
        @Override
        public Response intercept(Chain chain) throws IOException {
            //使用拦截器,添加公共参数
            Request request = chain.request();
            String method = request.method();
            if (method.equals("GET")) {
                String s = request.url().toString();
                s = s + "&source=android";
                Request newRequest = request.newBuilder().url(s).build();
                Response response = chain.proceed(newRequest);
                return response;

            } else if (method.equals("POST")) {
                RequestBody body = request.body();
                if (body instanceof FormBody) {
                    FormBody.Builder builder = new FormBody.Builder();
                    for (int i = 0; i < ((FormBody) body).size(); i++) {
                        String name = ((FormBody) body).name(i);
                        String value = ((FormBody) body).value(i);
                        builder.add(name, value);
                    }
                    builder.add("source", "android");
                    FormBody formBody = builder.build();
                    Request request1 = request.newBuilder().post(formBody).build();
                    Response response = chain.proceed(request1);
                    return response;
                }
            }
            return null;
        }
    }
    //handler
    class MyHandler extends Handler {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case 0:
                    //失败
                    String error = (String) msg.obj;
                    okLoadListener.okLoadError(error);
                    break;
                case 1:
                    //成功
                    String json = (String) msg.obj;
                    okLoadListener.okLoadSuccess(json);
                    break;
            }
        }
    }
    //提供外部可以访问的方法,用来设置监听的
    public void setOkLoadListener(OkLoadListener okLoadListener) {
        this.okLoadListener = okLoadListener;
    }
}

OkLoadListener

public interface OkLoadListener {
    void okLoadSuccess(String json);

    void okLoadError(String error);
}

model包下
GetGoodsListener

public interface GetGoodsListener {
    void getGoodsSuccess(String json);

    void getGoodsError(String error);
}

ModelImpl

import android.util.Log;
import com.example.shixun1_monthtest.http.OkHttpUitls;
import com.example.shixun1_monthtest.http.OkLoadListener;
public class ModelImpl implements IModel {
    private static final String TAG = "ModelImpl------";
    @Override
    public void getGoodsList(String url, final GetGoodsListener getGoodsListener) {
        //使用OkHttp请求
        OkHttpUitls okHttpUitls = OkHttpUitls.getOkHttpUitls();
        okHttpUitls.okGet(url);
        okHttpUitls.setOkLoadListener(new OkLoadListener() {
            @Override
            public void okLoadSuccess(String json) {
                Log.d(TAG, "okLoadSuccess: "+json);
                //将数据传入presenter
                getGoodsListener.getGoodsSuccess(json);
            }
            @Override
            public void okLoadError(String error) {
                getGoodsListener.getGoodsError(error);
            }
        });
    }
}

view包下
IMainView

import com.example.shixun1_monthtest.model.GoodsBean;
import java.util.List;
public interface IMainView {
    //显示商品列表的方法
    void showGoodsList(List<GoodsBean.DataBean> list);
}

DetailActivity

import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;
import android.widget.TextView;
import com.amap.api.location.AMapLocation;
import com.amap.api.location.AMapLocationClient;
import com.amap.api.location.AMapLocationClientOption;
import com.amap.api.location.AMapLocationListener;
import com.bumptech.glide.Glide;
import com.example.shixun1_monthtest.R;

public class DetailActivity extends AppCompatActivity {
    private static final String TAG = "DetailActivity====";
    private ImageView detail_pic;
    private TextView title;
    private TextView detail_address;
    public AMapLocationClient mLocationClient = null;
    //声明定位回调监听器
    //public AMapLocationListener mLocationListener = new AMapLocationListener();
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_detail);
        initViews();
        initDatas();
        initApi();
    }
    //-------------------------高德地图开始--------------------------------------
    private void initApi() {
        //声明AMapLocationClient类对象
        //初始化定位
        mLocationClient = new AMapLocationClient(getApplicationContext());
        //设置定位回调监听
        mLocationClient.setLocationListener(mAMapLocationListener);
        AMapLocationClientOption mLocationOption = null;
        //初始化AMapLocationClientOption对象
        mLocationOption = new AMapLocationClientOption();
        AMapLocationClientOption option = new AMapLocationClientOption();
        /**
         * 设置定位场景,目前支持三种场景(签到、出行、运动,默认无场景)
         */
        option.setLocationPurpose(AMapLocationClientOption.AMapLocationPurpose.SignIn);
        if (null != mLocationClient) {
            mLocationClient.setLocationOption(option);
            //设置场景模式后最好调用一次stop,再调用start以保证场景模式生效
            mLocationClient.stopLocation();
            mLocationClient.startLocation();
        }
        //定位模式
        mLocationOption.setLocationMode(AMapLocationClientOption.AMapLocationMode.Hight_Accuracy);
        //设置定位间隔,单位毫秒,默认为2000ms,最低1000ms。
        mLocationOption.setInterval(1000);
        //设置是否返回地址信息(默认返回地址信息)
        mLocationOption.setNeedAddress(true);
        //给定位客户端对象设置定位参数
        mLocationClient.setLocationOption(mLocationOption);
        //启动定位
        mLocationClient.startLocation();
    }

    AMapLocationListener mAMapLocationListener = new AMapLocationListener() {
        @Override
        public void onLocationChanged(AMapLocation amapLocation) {
            //获取数据的
            if (amapLocation != null) {
                if (amapLocation.getErrorCode() == 0) {
                    //可在其中解析amapLocation获取相应内容。
                    Log.d(TAG, "定位成功----");

                    String street = amapLocation.getStreet();
                    String address = amapLocation.getAddress();
                    Log.d(TAG, "onLocationChanged: "+street);
                    Log.d(TAG, "onLocationChanged: "+address);
                    //将地址显示出来
                    detail_address.setText(address);
                } else {
                    //定位失败时,可通过ErrCode(错误码)信息来确定失败的原因,errInfo是错误信息,详见错误码表。
                    Log.e("AmapError", "location Error, ErrCode:"
                            + amapLocation.getErrorCode() + ", errInfo:"
                            + amapLocation.getErrorInfo());
                }
            }
        }
    };
//-------------------------高德地图结束--------------------------------------

    private void initDatas() {
        Intent intent = getIntent();
        String pic_url = intent.getStringExtra("pic_url");
        String title1 = intent.getStringExtra("title");

        Glide.with(this).load(pic_url).into(detail_pic);
        title.setText(title1);
    }

    private void initViews() {
        detail_pic = findViewById(R.id.detail_pic);
        title = findViewById(R.id.detail_title);
        detail_address = findViewById(R.id.detail_address);
        //点击跳转
        detail_address.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(DetailActivity.this, MyAddressActivity.class);
                startActivity(intent);
            }
        });
    }
}

MyAddressActivity

package com.example.shixun1_monthtest.view;
import android.app.Activity;
import android.graphics.Color;
import android.os.Bundle;
import android.support.annotation.Nullable;
import com.amap.api.maps2d.AMap;
import com.amap.api.maps2d.MapView;
import com.amap.api.maps2d.model.MyLocationStyle;
import com.example.shixun1_monthtest.R;

public class MyAddressActivity extends Activity {
    private MapView mMapView;
    AMap aMap = null;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.address_layout);
        //获取地图控件引用
        mMapView = (MapView) findViewById(R.id.map);
        //在activity执行onCreate时执行mMapView.onCreate(savedInstanceState),创建地图
        mMapView.onCreate(savedInstanceState);
        //初始化地图控制器对象
        if (aMap == null) {
            aMap = mMapView.getMap();
        }
        initApi();
    }

    private void initApi() {
        MyLocationStyle myLocationStyle;
        myLocationStyle = new MyLocationStyle();//初始化定位蓝点样式类myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_LOCATION_ROTATE);//连续定位、且将视角移动到地图中心点,定位点依照设备方向旋转,并且会跟随设备移动。(1秒1次定位)如果不设置myLocationType,默认也会执行此种模式。
        myLocationStyle.interval(2000); //设置连续定位模式下的定位间隔,只在连续定位模式下生效,单次定位模式下不会生效。单位为毫秒。
        aMap.setMyLocationStyle(myLocationStyle);//设置定位蓝点的Style
        //aMap.getUiSettings().setMyLocationButtonEnabled(true);设置默认定位按钮是否显示,非必需设置。
        aMap.setMyLocationEnabled(true);// 设置为true表示启动显示定位蓝点,false表示隐藏定位蓝点并不进行定位,默认是false。
        myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_FOLLOW) ;//连续定位、且将视角移动到地图中心点,定位蓝点跟随设备移动。(1秒1次定位)
        myLocationStyle.showMyLocation(true);//显示蓝点
        myLocationStyle.strokeColor(Color.BLUE);//设置定位蓝点精度圆圈的边框颜色的方法。
        myLocationStyle.radiusFillColor(Color.RED);//
        myLocationStyle.strokeWidth(5);//设置定位蓝点精度圈的边框宽度的方法。
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        //在activity执行onDestroy时执行mMapView.onDestroy(),销毁地图
        mMapView.onDestroy();
    }

    @Override
    protected void onResume() {
        super.onResume();
        //在activity执行onResume时执行mMapView.onResume (),重新绘制加载地图
        mMapView.onResume();
    }

    @Override
    protected void onPause() {
        super.onPause();
        //在activity执行onPause时执行mMapView.onPause (),暂停地图的绘制
        mMapView.onPause();
    }

    @Override
    protected void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        //在activity执行onSaveInstanceState时执行mMapView.onSaveInstanceState (outState),保存地图当前的状态
        mMapView.onSaveInstanceState(outState);
    }
}











评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值