//activity_main
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.bwie.test.kaoshiti.MainActivity"> <com.bwie.test.view.CustomBanner android:id="@+id/custom_banner" android:layout_width="match_parent" android:layout_height="wrap_content"> </com.bwie.test.view.CustomBanner> <Button android:layout_width="match_parent" android:layout_height="50dp" android:background="@drawable/btnbg" android:text="按钮"/> </LinearLayout>
//activity_details
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.bwie.test.view.DetailsActivity"> <WebView android:id="@+id/webView" android:layout_width="match_parent" android:layout_height="match_parent"> </WebView> </RelativeLayout>
//banner_layout
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <android.support.v4.view.ViewPager android:id="@+id/banner_view_pager" android:layout_width="match_parent" android:layout_height="200dp"> </android.support.v4.view.ViewPager> <LinearLayout android:id="@+id/linear_bannner" android:layout_centerHorizontal="true" android:layout_alignBottom="@+id/banner_view_pager" android:layout_marginBottom="10dp" android:orientation="horizontal" android:layout_width="wrap_content" android:layout_height="wrap_content"> </LinearLayout> </RelativeLayout>
drawable文件夹下//btnbg
<?xml version="1.0" encoding="utf-8"?> <shape xmlns:android="http://schemas.android.com/apk/res/android" android:shape="rectangle"> <corners android:radius="20dp"></corners> <gradient android:startColor="#f00" android:endColor="#f89" android:angle="45"/> </shape>
//shape_01
<?xml version="1.0" encoding="utf-8"?> <shape xmlns:android="http://schemas.android.com/apk/res/android"> <!-- 颜色 --> <solid android:color="#f89"></solid> <!-- 圆角 --> <corners android:radius="10dp"></corners> <!-- 大小 --> <size android:height="10dp" android:width="10dp"></size> </shape>
//shape_02
<?xml version="1.0" encoding="utf-8"?> <shape xmlns:android="http://schemas.android.com/apk/res/android"> <!-- 颜色 --> <solid android:color="#f00"></solid> <!-- 圆角 --> <corners android:radius="10dp"></corners> <!-- 大小 --> <size android:height="10dp" android:width="10dp"></size> </shape>
//MainActivity
public class MainActivity extends AppCompatActivity { private CustomBanner banner; List<String> list = new ArrayList<>(); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); //需求 实现与banner一样的轮播图 banner = (CustomBanner) findViewById(R.id.custom_banner); //请求数据进行解析展示 getDataFromNet(); } public void getDataFromNet() { OkHttp3Util.doGet("http://120.27.23.105/ad/getAd", new Callback() { @Override public void onFailure(Call call, IOException e) { } @Override public void onResponse(Call call, Response response) throws IOException { if (response.isSuccessful()) { String json = response.body().string(); final DetalBean detalBean = new Gson().fromJson(json, DetalBean.class); //因为图片的连接在一起,所以要截断 // String[] images = detalBean.getData().getImages().split("\\|"); list = new ArrayList<>(); for (int i = 0; i < detalBean.getData().size(); i++) { // list.add(images[i]); if (detalBean.getData().get(i).getIcon() != null) { // 图片添加到集合 list.add(detalBean.getData().get(i).getIcon()); } } runOnUiThread(new Runnable() { @Override public void run() { //设置显示轮播 banner.setImageUrls(list); //设置点击事件 banner.setClickListner(new CustomBanner.OnClickLisner() { @Override public void onItemClick(int position) { if (detalBean.getData().get(position).getType() == 0) { //跳转到详情页面 Intent intent = new Intent(MainActivity.this, DetailsActivity.class); intent.putExtra("url", detalBean.getData().get(position).getUrl()); startActivity(intent); } else { Toast.makeText(MainActivity.this, "点击" + position, Toast.LENGTH_SHORT).show(); } } }); } }); } } }); } }
自定义View//DetailsActivity
public class DetailsActivity extends AppCompatActivity { private WebView webView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_details); webView = (WebView) findViewById(R.id.webView); Intent intent = getIntent(); //接收传过来的值 String url = intent.getStringExtra("url"); //加载 webView.loadUrl(url);//加载网址到视图 //设置 webView.setWebViewClient(new WebViewClient()); WebSettings settings = webView.getSettings(); settings.setJavaScriptEnabled(true); settings.setJavaScriptCanOpenWindowsAutomatically(true); } }
//CustomBanner
public class CustomBanner extends FrameLayout { private ViewPager viewPager; private LinearLayout linearLayout; private List<String> list; int time = 2; //消息接受 Handler hander = new Handler() { @Override public void handleMessage(Message msg) { super.handleMessage(msg); if (msg.what == 0) { int currentItem = viewPager.getCurrentItem(); viewPager.setCurrentItem(currentItem + 1); //再次发送 sendEmptyMessageDelayed(0, time * 1000); } } }; private ArrayList<ImageView> listDoc; private OnClickLisner onClickLisner; //创建类时,自动创建了三个构造方法,当调用本类时,都会走构造方法里的方法,也就是初始化 public CustomBanner(@NonNull Context context) { super(context); init(); } public CustomBanner(@NonNull Context context, @Nullable AttributeSet attrs) { super(context, attrs); init(); } public CustomBanner(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); } //初始化方法 private void init() { //找到banner布局 View view = View.inflate(getContext(), R.layout.banner_layout, this); //找到控件 viewPager = view.findViewById(R.id.banner_view_pager); linearLayout = view.findViewById(R.id.linear_bannner); } /** * 对外提供设置image路径的方法 * 当我调用这个方法的时候,给它传一个集合,集合中装的是要展示的图片 */ public void setImageUrls(List<String> list) { this.list = list; //当集合为空的时候,不至于报错 if (list == null) { return; } //设置适配器 当我获取到数据的时候,我要让它首先能展示图片,然后在做手动滑动,最后实现无线自动轮播 LunBoAdapter lunBoAdapter = new LunBoAdapter(getContext(), list); viewPager.setAdapter(lunBoAdapter); //加载小圆点 initDoc(); //显示中间某个位置 就是初始化的时候,可以让他来回翻动 viewPager.setCurrentItem(list.size() * 10000); //使用handler自动轮播 hander.sendEmptyMessageDelayed(0, time * 1000); viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() { @Override /* * 当用手指滑动翻页的时候,如果翻动成功了(滑动的距离够长), * 手指抬起来就会立即执行这个方法,position就是当前滑动到的页面。如果直接setCurrentItem翻页,那position就和setCurrentItem的参数一致, * 这种情况在onPageScrolled执行方法前就会立即执行 * */ public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) { //在选中某一页的时候,切换小圆点的背景 for (int i = 0; i < listDoc.size(); i++) { if (position % listDoc.size() == i) { listDoc.get(i).setBackgroundResource(R.drawable.shape_01); } else { listDoc.get(i).setBackgroundResource(R.drawable.shape_02); } } } @Override public void onPageSelected(int position) { } @Override public void onPageScrollStateChanged(int state) { } }); } /** * 点击事件 * @param onClickLisner */ public void setClickListner(OnClickLisner onClickLisner) { this.onClickLisner = onClickLisner; } private class LunBoAdapter extends PagerAdapter { private List<String> list; private Context context; public LunBoAdapter(Context context, List<String> list) { this.context = context; this.list = list; } //我要重写两个方法,才能实现功能 @Override public int getCount() { return Integer.MAX_VALUE; } @Override public boolean isViewFromObject(View view, Object object) { return view == object; } //加载 当页面滑动时,预加载左右两个图片 @Override public Object instantiateItem(ViewGroup container, final int position) { //创建imageView ImageView imageView = new ImageView(getContext()); //铺满横屏 imageView.setScaleType(ImageView.ScaleType.FIT_XY); //加载这张图片 Glide.with(getContext()).load(list.get(position % list.size())).into(imageView); //点击事件 imageView.setOnClickListener(new OnClickListener() { @Override public void onClick(View view) { //触发 onClickLisner.onItemClick(position%list.size()); } }); //添加到容器 container.addView(imageView); //返回 return imageView; } //销毁 @Override public void destroyItem(ViewGroup container, int position, Object object) { container.removeView((View) object); } } private void initDoc() { //创建一个集合,记录这些小圆点 listDoc = new ArrayList<ImageView>(); //清空布局 // linearLayout.removeAllViews(); for (int i = 0; i < list.size(); i++) { //创建一个图片的占位 ImageView docImage = new ImageView(getContext()); if (i == 0) { //默认小圆点第一个显示带颜色 docImage.setBackgroundResource(R.drawable.shape_01); } else { //其他变成平常的 颜色 docImage.setBackgroundResource(R.drawable.shape_02); } //添加到集合 设置完小圆点添加到集合 可能是每次设置完之后都添加再展示 listDoc.add(docImage); //添加到线性布局 //第一个参数为宽的设置,第二个参数为高的设置。 //(用的时候注意修改LinearLayout前缀,其实很多时候可以不写它的) LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT); //设置 设置MarginS margin与padding 设置边距的 params.setMargins(5, 0, 5, 0); //整个小圆点的父布局添加布局图片的控件 第二个参数是设置位置 linearLayout.addView(docImage, params); } } //内部接口 public interface OnClickLisner{ void onItemClick(int position); } }
util文件夹下//OkHttp3Util
public class OkHttp3Util { private static OkHttpClient okHttpClient = null; private OkHttp3Util() { } public static OkHttpClient getInstance() { if (okHttpClient == null) { //加同步安全 synchronized (OkHttp3Util.class) { if (okHttpClient == null) { //okhttp可以缓存数据....指定缓存路径 File sdcache = new File(Environment.getExternalStorageDirectory(), "cache"); //指定缓存大小 int cacheSize = 10 * 1024 * 1024; okHttpClient = new OkHttpClient.Builder()//构建器 .connectTimeout(15, TimeUnit.SECONDS)//连接超时 .writeTimeout(20, TimeUnit.SECONDS)//写入超时 .readTimeout(20, TimeUnit.SECONDS)//读取超时 .cache(new Cache(sdcache.getAbsoluteFile(), cacheSize))//设置缓存 .build(); } } } return okHttpClient; } /** * get请求 * 参数1 url * 参数2 回调Callback */ public static void doGet(String oldUrl, Callback callback) { //创建OkHttpClient请求对象 OkHttpClient okHttpClient = getInstance(); //创建Request Request request = new Request.Builder().url(oldUrl).build(); //得到Call对象 Call call = okHttpClient.newCall(request); //执行异步请求 call.enqueue(callback); } /** * post请求 * 参数1 url * 参数2 Map<String, String> params post请求的时候给服务器传的数据 * add..("","") * add() */ public static void doPost(String url, Map<String, String> params, Callback callback) { //创建OkHttpClient请求对象 OkHttpClient okHttpClient = getInstance(); //3.x版本post请求换成FormBody 封装键值对参数 FormBody.Builder builder = new FormBody.Builder(); //遍历集合,,,map集合遍历方式 for (String key : params.keySet()) { builder.add(key, params.get(key)); } //创建Request....formBody...new formBody.Builder()...add()....build() Request request = new Request.Builder().url(url).post(builder.build()).build(); Call call = okHttpClient.newCall(request); call.enqueue(callback); } /** * post请求上传文件....包括图片....流的形式传任意文件... * 参数1 url * file表示上传的文件 * fileName....文件的名字,,例如aaa.jpg * params ....传递除了file文件 其他的参数放到map集合 */ public static void uploadFile(String url, File file, String fileName,Map<String,String> params,Callback callback) { //创建OkHttpClient请求对象 OkHttpClient okHttpClient = getInstance(); //MultipartBody多功能的请求实体对象,,,formBody只能传表单形式的数据 MultipartBody.Builder builder = new MultipartBody.Builder(); builder.setType(MultipartBody.FORM); //参数 if (params != null){ for (String key : params.keySet()){ builder.addFormDataPart(key,params.get(key)); } } //文件...参数name指的是请求路径中所接受的参数...如果路径接收参数键值是fileeeee,此处应该改变 builder.addFormDataPart("file",fileName,RequestBody.create(MediaType.parse("application/octet-stream"),file)); //构建 MultipartBody multipartBody = builder.build(); //创建Request Request request = new Request.Builder().url(url).post(multipartBody).build(); //得到Call Call call = okHttpClient.newCall(request); //执行请求 call.enqueue(callback); } /** * Post请求发送JSON数据....{"name":"zhangsan","pwd":"123456"} * 参数一:请求Url * 参数二:请求的JSON * 参数三:请求回调 */ public static void doPostJson(String url, String jsonParams, Callback callback) { RequestBody requestBody = RequestBody.create(MediaType.parse("application/json; charset=utf-8"), jsonParams); Request request = new Request.Builder().url(url).post(requestBody).build(); Call call = getInstance().newCall(request); call.enqueue(callback); } /** * 下载文件 以流的形式把apk写入的指定文件 得到file后进行安装 * 参数er:请求Url * 参数san:保存文件的文件夹....download */ public static void download(final Activity context, final String url, final String saveDir) { Request request = new Request.Builder().url(url).build(); Call call = getInstance().newCall(request); call.enqueue(new Callback() { @Override public void onFailure(Call call, IOException e) { //com.orhanobut.logger.Logger.e(e.getLocalizedMessage()); } @Override public void onResponse(Call call, final Response response) throws IOException { InputStream is = null; byte[] buf = new byte[2048]; int len = 0; FileOutputStream fos = null; try { is = response.body().byteStream();//以字节流的形式拿回响应实体内容 //apk保存路径 final String fileDir = isExistDir(saveDir); //文件 File file = new File(fileDir, getNameFromUrl(url)); fos = new FileOutputStream(file); while ((len = is.read(buf)) != -1) { fos.write(buf, 0, len); } fos.flush(); context.runOnUiThread(new Runnable() { @Override public void run() { Toast.makeText(context, "下载成功:" + fileDir + "," + getNameFromUrl(url), Toast.LENGTH_SHORT).show(); } }); //apk下载完成后 调用系统的安装方法 Intent intent = new Intent(Intent.ACTION_VIEW); intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive"); context.startActivity(intent); } catch (IOException e) { e.printStackTrace(); } finally { if (is != null) is.close(); if (fos != null) fos.close(); } } }); } /** * 判断下载目录是否存在......并返回绝对路径 * * @param saveDir * @return * @throws IOException */ public static String isExistDir(String saveDir) throws IOException { // 下载位置 if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) { File downloadFile = new File(Environment.getExternalStorageDirectory(), saveDir); if (!downloadFile.mkdirs()) { downloadFile.createNewFile(); } String savePath = downloadFile.getAbsolutePath(); Log.e("savePath", savePath); return savePath; } return null; } /** * @param url * @return 从下载连接中解析出文件名 */ private static String getNameFromUrl(String url) { return url.substring(url.lastIndexOf("/") + 1); } }
//添加的依赖
compile 'com.google.code.gson:gson:2.8.2' compile 'com.nostra13.universalimageloader:universal-image-loader:1.9.5' compile 'com.android.support.test:runner:0.5' compile 'com.android.support.test.espresso:espresso-core:2.2.2' compile 'com.squareup.okhttp3:okhttp:3.6.0' compile 'com.squareup.okio:okio:1.11.0' compile files('libs/glide-3.7.0.jar')
//添加的权限
<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"/>