主方法类
package com.dash.a03_custom_view_02;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.Toast;
import com.dash.a03_custom_view_02.bean.DetalBean;
import com.dash.a03_custom_view_02.util.OkHttp3Util;
import com.dash.a03_custom_view_02.view.CombineView;
import com.dash.a03_custom_view_02.view.CustomBanner;
import com.google.gson.Gson;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import okhttp3.Call;
import okhttp3.Callback;
import okhttp3.Response;
public class MainActivity extends AppCompatActivity {
private CustomBanner customBanner;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
customBanner = findViewById(R.id.custom_banner);
getDataFromNet();
}
private void getDataFromNet() {
OkHttp3Util.doGet("https://www.zhaoapi.cn/product/getProductDetail?Pid=1&source=android", 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);
final List<String> list = new ArrayList<>();
String[] images = detalBean.getData().getImages().split("\\|");
for (int i = 0;i<images.length;i++){
list.add(images[i]);
}
runOnUiThread(new Runnable() {
@Override
public void run() {
customBanner.setTimeSecond(5);
customBanner.setImageUrls(list);
customBanner.setClickListner(new CustomBanner.OnClickLisner() {
@Override
public void onItemClick(int position) {
Toast.makeText(MainActivity.this,"点击",Toast.LENGTH_SHORT).show();
}
});
}
});
}
}
});
}
}
CombineView 类
import android.content.Context;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.view.View;
import android.widget.CheckBox;
import android.widget.FrameLayout;
import android.widget.TextView;
import com.dash.a03_custom_view_02.R;
/**
* Created by Dash on 2017/12/28.
*/
public class CombineView extends FrameLayout implements View.OnClickListener {
private TextView textView;
private CheckBox checkBox;
private String text;
private boolean checked;
public CombineView(@NonNull Context context) {
super(context);
init();
}
public CombineView(@NonNull Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
text = attrs.getAttributeValue("http://schemas.android.com/apk/res-auto", "text");
checked = attrs.getAttributeBooleanValue("http://schemas.android.com/apk/res-auto", "checked", false);
init();
}
public CombineView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
/**
* 初始化的方法....加载出布局,并且添加到当前自定义view上
*/
private void init() {
View view = View.inflate(getContext(), R.layout.combine_layout, this);
textView = view.findViewById(R.id.combine_text);
checkBox = view.findViewById(R.id.combine_check);
this.setOnClickListener(this);
textView.setText(text);
checkBox.setChecked(checked);
}
/**
* 对外提供设置文本的方法
*/
public void setText(String text){
textView.setText(text);
}
/**
* 对外提供设置是否选中的方法
*/
public void setChecked(boolean flag){
checkBox.setChecked(flag);
}
/**
* 对外提供是否选中的方法
*/
public boolean getChecked(){
return checkBox.isChecked();
}
@Override
public void onClick(View view) {
checkBox.setChecked(! checkBox.isChecked());
}
}
CountView类
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.view.View;
/**
* Created by Dash on 2017/12/28.
*/
public class CountView extends View implements View.OnClickListener {
private int count = 0;
public CountView(Context context) {
super(context);
init();
}
public CountView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
init();
}
public CountView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
private void init() {
this.setOnClickListener(this);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
Paint paint = new Paint();
paint.setColor(Color.RED);
paint.setAntiAlias(true);
paint.setStyle(Paint.Style.FILL);
canvas.drawCircle(300,300,200,paint);
paint.setColor(Color.BLACK);
paint.setTextSize(100);
String text = String.valueOf(count);
Rect rect = new Rect();
paint.getTextBounds(text,0,text.length(),rect);
canvas.drawText(text,300-rect.width()/2,300+rect.height()/2,paint);
}
@Override
public void onClick(View view) {
count ++;
postInvalidate();
}
}
自定义Banner类
import android.content.Context;
import android.os.Handler;
import android.os.Message;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.v4.view.PagerAdapter;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.LinearLayout;
import com.bumptech.glide.Glide;
import com.dash.a03_custom_view_02.R;
import java.util.ArrayList;
import java.util.List;
/**
* Created by Dash on 2017/12/29.
*/
public class CustomBanner extends FrameLayout {
private ViewPager viewPager;
private LinearLayout linearLayout;
private List<String> list;
private int time = 2;
private Handler handler = new Handler(){
@Override
public void handleMessage(Message msg) {
if (msg.what == 0){
int currentItem = viewPager.getCurrentItem();
viewPager.setCurrentItem(currentItem +1);
sendEmptyMessageDelayed(0,time*1000);
}
}
};
private List<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() {
View view = View.inflate(getContext(), R.layout.bannner_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.sendEmptyMessageDelayed(0,time*1000);
viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
}
@Override
public void onPageSelected(int position) {
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 onPageScrollStateChanged(int state) {
}
});
}
/**
* 初始化小圆点
*/
private void initDoc() {
listDoc = new ArrayList<>();
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.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT,LinearLayout.LayoutParams.WRAP_CONTENT);
params.setMargins(5,0,5,0);
linearLayout.addView(docImage,params);
}
}
/**
* 对外提供轮播的时间
*/
public void setTimeSecond(int time){
this.time = time;
}
/**
* 点击事件
* @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 = new ImageView(context);
imageView.setScaleType(ImageView.ScaleType.FIT_XY);
Glide.with(context).load(list.get(position%list.size())).into(imageView);
imageView.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View view) {
onClickLisner.onItemClick(position%list.size());
}
});
imageView.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View view, MotionEvent motionEvent) {
switch (motionEvent.getAction()){
case MotionEvent.ACTION_DOWN:
handler.removeCallbacksAndMessages(null);
break;
case MotionEvent.ACTION_MOVE:
handler.removeCallbacksAndMessages(null);
break;
case MotionEvent.ACTION_CANCEL:
handler.sendEmptyMessageDelayed(0,time*1000);
break;
case MotionEvent.ACTION_UP:
handler.sendEmptyMessageDelayed(0,time*1000);
break;
}
return false;
}
});
container.addView(imageView);
return imageView;
}
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
container.removeView((View) object);
}
}
public interface OnClickLisner{
void onItemClick(int position);
}
}
OK封装
import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Environment;
import android.util.Log;
import android.widget.Toast;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Map;
import java.util.concurrent.TimeUnit;
import okhttp3.Cache;
import okhttp3.Call;
import okhttp3.Callback;
import okhttp3.FormBody;
import okhttp3.MediaType;
import okhttp3.MultipartBody;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.RequestBody;
import okhttp3.Response;
/**
* Created by Dash on 2017/12/8.
*/
public class OkHttp3Util {
/**
* 懒汉 安全 加同步
* 1.私有的静态成员变量 只声明不创建
* 2.私有的构造方法
* 3.提供返回实例的静态方法
*/
private static OkHttpClient okHttpClient = null;
private OkHttp3Util() {
}
public static OkHttpClient getInstance() {
if (okHttpClient == null) {
synchronized (OkHttp3Util.class) {
if (okHttpClient == null) {
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 = getInstance();
Request request = new Request.Builder().url(oldUrl).build();
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 = getInstance();
FormBody.Builder builder = new FormBody.Builder();
for (String key : params.keySet()) {
builder.add(key, params.get(key));
}
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 = getInstance();
MultipartBody.Builder builder = new MultipartBody.Builder();
builder.setType(MultipartBody.FORM);
if (params != null){
for (String key : params.keySet()){
builder.addFormDataPart(key,params.get(key));
}
}
builder.addFormDataPart("file",fileName,RequestBody.create(MediaType.parse("application/octet-stream"),file));
MultipartBody multipartBody = builder.build();
Request request = new Request.Builder().url(url).post(multipartBody).build();
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) {
}
@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();
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();
}
});
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);
}
}
主方法布局
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:dash="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.dash.a03_custom_view_02.MainActivity">
<com.dash.a03_custom_view_02.view.CustomBanner
android:id="@+id/custom_banner"
android:layout_width="match_parent"
android:layout_height="wrap_content">
</com.dash.a03_custom_view_02.view.CustomBanner>
</RelativeLayout>
自定义banner布局
<?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="wrap_content">
<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>
<?xml version="1.0" encoding="UTF-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<solid android:color="#00ff00"/>
<corners android:radius="10dp"/>
<size android:width="10dp" android:height="10dp"/>
</shape>
<?xml version="1.0" encoding="UTF-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<solid android:color="#ff0000"/>
<corners android:radius="10dp"/>
<size android:width="10dp" android:height="10dp"/>
</shape>