HorizontalScrollView Horizontal水平滑动栏和今日头条ViewPager的联动

一.  传值的 MainActivity.java

package com.Horizaontal;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentPagerAdapter;
import android.support.v4.view.ViewPager;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

public class MainActivity extends AppCompatActivity {
    private ViewPager viewPager;
    private MyIndicator myIndicator;
    private String[] titles;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        viewPager = (ViewPager) findViewById(R.id.viewPager);
        myIndicator = (MyIndicator) findViewById(R.id.indicator);

        //1.    设置指示器标题
        titles = new  String[]{"头条","社会","国内","军事","娱乐","科技","时尚","财经","体育"};
        myIndicator.setTitles(titles);
        viewPager.setOffscreenPageLimit(titles.length);

        //2.    为ViewPager设置适配器
        viewPager.setAdapter(new FragmentPagerAdapter(getSupportFragmentManager()) {
            @Override
            public CharSequence getPageTitle(int position) {
                return titles[position];
            }

            @Override
            public Fragment getItem(int position) {
                //获取标题,向Fragment传值
                News_Fragment news_fragment = new News_Fragment();
                Bundle bundle = new Bundle();
                if (titles[position].equals("头条")){
                    bundle.putString("name","top");
                }else if  (titles[position].equals("社会")){
                    bundle.putString("name","shehui");
                }else if  (titles[position].equals("国内")){
                    bundle.putString("name","guonei");
                }else if  (titles[position].equals("军事")){
                    bundle.putString("name","junshi");
                }else if  (titles[position].equals("娱乐")){
                    bundle.putString("name","yule");
                }else if  (titles[position].equals("科技")){
                    bundle.putString("name","keji");
                }else if  (titles[position].equals("时尚")){
                    bundle.putString("name","shishang");
                }else if  (titles[position].equals("财经")){
                    bundle.putString("name","caijing");
                }else if  (titles[position].equals("体育")){
                    bundle.putString("name","tiyu");
                }
                news_fragment.setArguments(bundle);
                return news_fragment;
            }

            @Override
            public int getCount() {
                return titles.length;
            }
        });

        //3.    指示器要与ViewPager关联
        myIndicator.setViewPager(viewPager);
    }
}

二.  设置多条目展示的适配器

package com.Horizaontal;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.TextView;
import com.nostra13.universalimageloader.core.ImageLoader;
import java.util.List;
import util.ImageLoaderUtil;

public class NewsAdapter extends BaseAdapter{
    private List<JsonBean.ResultBean.DataBean> list;
    private Context context;
    private int ONLY_TITLE = 0;
    private int IMAGE_ONE = 1;
    private int IMAGE_TWO = 2;
    private int IMAGE_THREE = 3;

    public NewsAdapter(Context context, List<JsonBean.ResultBean.DataBean> list) {
        this.context = context;
        this.list = list;
    }

    @Override
    public int getCount() {
        return list.size();
    }

    @Override
    public Object getItem(int position) {
        return list.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public int getViewTypeCount() {
        return 4;
    }

    @Override
    public int getItemViewType(int position) {
        /**每个fragment中展示新闻内容,要求:多条目加载和上拉加载,下拉刷新
            多条目加载:奇数条目(第1357....)展示只有标题的布局
            偶数条目(第2468.....)展示标题+图片样式的条目布局
         */
        if (position % 2 == 0){
            if (list.get(position).getThumbnail_pic_s() != null && list.get(position).getThumbnail_pic_s02() != null &&list.get(position).getThumbnail_pic_s03() != null ){
                return IMAGE_THREE;
            }else if (list.get(position).getThumbnail_pic_s() != null && list.get(position).getThumbnail_pic_s02() != null){
                return IMAGE_TWO;
            }
            return IMAGE_ONE;
        }
        return ONLY_TITLE;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if (getItemViewType(position) == IMAGE_ONE){
            Image01_ViewHolder holder;
            if (convertView == null){
                convertView =View.inflate(context, R.layout.item_layout01,null);
                holder =new Image01_ViewHolder();

                //查找控件
                holder.author_name = (TextView) convertView.findViewById(R.id.author_name);
                holder.title = (TextView) convertView.findViewById(R.id.title);
                holder.image = (ImageView) convertView.findViewById(R.id.image);
                convertView.setTag(holder);
            }else {
                holder = (Image01_ViewHolder) convertView.getTag();
            }

            //获取数据重新赋值
            holder.title.setText(list.get(position).getTitle());
            holder.author_name.setText(list.get(position).getAuthor_name());
            ImageLoader.getInstance().displayImage(list.get(position).getThumbnail_pic_s(),holder.image, ImageLoaderUtil.getDefaultOption());
        }else if (getItemViewType(position) == IMAGE_TWO){
            Image02_ViewHolder holder;
            if (convertView == null){
                convertView =View.inflate(context, R.layout.item_layout02,null);
                holder =new Image02_ViewHolder();

                //查找控件
                holder.image002 = (ImageView) convertView.findViewById(R.id.image002);
                holder.image001 = (ImageView) convertView.findViewById(R.id.image001);
                holder.title = (TextView) convertView.findViewById(R.id.title);
                convertView.setTag(holder);
            }else {
                holder = (Image02_ViewHolder) convertView.getTag();
            }

            //获取数据重新赋值
            holder.title.setText(list.get(position).getTitle());
            ImageLoader.getInstance().displayImage(list.get(position).getThumbnail_pic_s(),holder.image001,ImageLoaderUtil.getDefaultOption());
            ImageLoader.getInstance().displayImage(list.get(position).getThumbnail_pic_s02(),holder.image002,ImageLoaderUtil.getDefaultOption());
        } else if (getItemViewType(position) == IMAGE_THREE){
            Image03_ViewHolder holder;
            if (convertView == null){
                convertView =View.inflate(context, R.layout.item_layout03,null);
                holder =new Image03_ViewHolder();

                //查找控件
                holder.image01 = (ImageView) convertView.findViewById(R.id.image01);
                holder.image02 = (ImageView) convertView.findViewById(R.id.image02);
                holder.image03 = (ImageView) convertView.findViewById(R.id.image03);
                holder.title = (TextView) convertView.findViewById(R.id.title);
                convertView.setTag(holder);
            }else {
                holder = (Image03_ViewHolder) convertView.getTag();
            }

            //获取数据重新赋值
            holder.title.setText(list.get(position).getTitle());
            ImageLoader.getInstance().displayImage(list.get(position).getThumbnail_pic_s(),holder.image01, ImageLoaderUtil.getDefaultOption());
            ImageLoader.getInstance().displayImage(list.get(position).getThumbnail_pic_s02(),holder.image02,ImageLoaderUtil.getDefaultOption());
            ImageLoader.getInstance().displayImage(list.get(position).getThumbnail_pic_s03(),holder.image03,ImageLoaderUtil.getDefaultOption());
        }else {
            Only_Title_ViewHolder holder;
            if (convertView == null){
                convertView =View.inflate(context, R.layout.title_layout,null);
                holder =new Only_Title_ViewHolder();

                //查找控件
                holder.title = (TextView) convertView.findViewById(R.id.title);
                convertView.setTag(holder);
            }else {
                holder = (Only_Title_ViewHolder) convertView.getTag();
            }

            //获取数据重新赋值
            holder.title.setText(list.get(position).getTitle());
        }
        return convertView;
    }

    //自定义优化缓存类
    static  class  Only_Title_ViewHolder{
        TextView title;
    }
    static  class  Image01_ViewHolder{
        TextView title,author_name;
        ImageView image;
    }
    static  class  Image02_ViewHolder{
        TextView title;
        ImageView image001,image002;
    }
    static  class  Image03_ViewHolder{
        TextView title;
        ImageView image01,image02,image03;
    }
}

三. 自定义的指示器

package com.Horizaontal;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.view.Gravity;
import android.view.View;
import android.widget.HorizontalScrollView;
import android.widget.LinearLayout;
import android.widget.TextView;

public class MyIndicator extends HorizontalScrollView implements ViewPager.OnPageChangeListener{

    private static final int COLOR_TEXT_NORMAL = 0xFF000000;
    private static final int COLOR_INDICATOR_COLOR = Color.BLACK;

    private Context context;
    private  int tabWidth;
    private String[] titles;
    private int count;
    private Paint mPaint;
    private float mTranslationX;
    private ViewPager viewPager;
    private int SCREEN_WIDTH;
    private float lineheight = 2.0f;

    public MyIndicator(Context context) {
        this(context, null);
    }

    public MyIndicator(Context context, AttributeSet attrs) {
        super(context, attrs);
        init(context);
    }

    public MyIndicator(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init(context);
    }

    private void init(Context context){
        this.context = context;
        mPaint = new Paint();
        mPaint.setColor(COLOR_INDICATOR_COLOR);
        mPaint.setStrokeWidth(lineheight);//底部指示线的宽度
        setHorizontalScrollBarEnabled(false);
        SCREEN_WIDTH = context.getResources().getDisplayMetrics().widthPixels;
    }

    public void setLineheight(float height){
        this.lineheight = height;
        mPaint.setStrokeWidth(lineheight);//底部指示线的宽度
    }

    public void setViewPager(ViewPager viewPager){
        this.viewPager = viewPager;
        viewPager.addOnPageChangeListener(this);
    }

    public void setTitles(String[] titles){
        this.titles = titles;
        count = titles.length;
        tabWidth = SCREEN_WIDTH/4;
        generateTitleView();
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        tabWidth = SCREEN_WIDTH/4;
    }

    @Override
    protected void dispatchDraw(Canvas canvas)
    {
        super.dispatchDraw(canvas);
        canvas.save();
        canvas.translate(mTranslationX, getHeight() - lineheight);
        canvas.drawLine(0, 0, tabWidth, 0, mPaint);//(startX, startY, stopX, stopY, paint)
        canvas.restore();
    }

    public void scroll(int position, float offset)
    {
        mTranslationX = tabWidth * (position + offset);
        scrollTo((int)mTranslationX-(SCREEN_WIDTH-tabWidth)/2, 0);
        invalidate();
    }

    private void generateTitleView()
    {
        if (getChildCount() > 0)
            this.removeAllViews();
        count = titles.length;

        LinearLayout linearLayout = new LinearLayout(context);
        linearLayout.setOrientation(LinearLayout.HORIZONTAL);
        linearLayout.setLayoutParams(new LinearLayout.LayoutParams(count*tabWidth, LinearLayout.LayoutParams.MATCH_PARENT));
        for (int i = 0; i < count; i++)
        {
            TextView tv = new TextView(getContext());
            LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(tabWidth,
                    LinearLayout.LayoutParams.MATCH_PARENT);
            tv.setGravity(Gravity.CENTER);
            tv.setTextColor(COLOR_TEXT_NORMAL);
            tv.setText(titles[i]);
            tv.setTextSize(TypedValue.COMPLEX_UNIT_SP, 16);//字体大小
            tv.setLayoutParams(lp);
            final int finalI = i;
            tv.setOnClickListener(new OnClickListener()
            {
                @Override
                public void onClick(View v)
                {
                    if(viewPager!=null){
                        viewPager.setCurrentItem(finalI,false);
                    }
                }
            });
            linearLayout.addView(tv);
        }
        addView(linearLayout);
    }

    @Override
    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
        scroll(position, positionOffset);
    }

    @Override
    public void onPageSelected(int position) {

    }

    @Override
    public void onPageScrollStateChanged(int state) {

    }
}

四. 接收数据的Fragment

package com.Horizaontal;
import android.os.AsyncTask;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import com.google.gson.Gson;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import util.NetConnectionUtil;
import util.StringUtil;
import view.XListView;

public class News_Fragment extends Fragment implements XListView.IXListViewListener{
    private List<JsonBean.ResultBean.DataBean> list = new ArrayList<>();
    private NewsAdapter adapter;
    private XListView xListView;
    private String data;

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.news_layout, container, false);

        //查找控件
        xListView = (XListView) view.findViewById(R.id.XListView);

        //设置XListView的上拉刷新功能
        xListView.setPullLoadEnable(true);
        xListView.setPullRefreshEnable(true);

        //XListView的监听事件
        xListView.setXListViewListener(this);
        return view;
    }

    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);

        //获取传递的标题数据
        Bundle bundle = getArguments();
        data = bundle.getString("name", "top");

        //判断网络状态,异步加载数据
        if (NetConnectionUtil.isNetConnectioned(getActivity())){
            getDataFromNet(data);
        }else{
            NetConnectionUtil.setNetConnectionWork(getActivity());
        }

    }

    private void getDataFromNet(final String data) {
        AsyncTask<Void,Void,String> task = new AsyncTask<Void, Void, String>() {
            @Override
            protected String doInBackground(Void... params) {
                try {
                    //路径
                    String path = "http://v.juhe.cn/toutiao/index?type="+data+"&key=c4479ad58f41e7f78a8fa073d0b1f1b5";
                    //连接网络
                    URL url = new URL(path);
                    HttpURLConnection connection = (HttpURLConnection) url.openConnection();
                    connection.setRequestMethod("GET");
                    connection.setReadTimeout(5000);
                    connection.setConnectTimeout(5000);
                    //响应数据
                    int responseCode = connection.getResponseCode();
                    if (responseCode == 200){
                        InputStream inputStream = connection.getInputStream();
                        String json = StringUtil.streamToString(inputStream,"utf-8");
                        return  json;
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
                return "";
            }

            @Override
            protected void onPostExecute(String json) {
                if (json== null || json.isEmpty()){

                }else {
                    //解析数据
                    JsonBean jsonBean = new Gson().fromJson(json, JsonBean.class);

                    if (jsonBean != null && jsonBean.getResult().getData() != null){

                        //把获取的数据添加到集合中
                        list.addAll(jsonBean.getResult().getData());

                        //设置适配器
                        setAdapter();
                        //停止刷新数据
                        xListView.stopRefresh();

                        //设置刷新的时间
                        Date date = new Date(System.currentTimeMillis());
                        SimpleDateFormat format = new SimpleDateFormat("HH:mm");
                        xListView.setRefreshTime(format.format(date));
                    }
                }
            }
        };
        task.execute();
    }

    //设置适配器的方法
    private void setAdapter() {
        if (adapter == null){
            adapter = new NewsAdapter(getActivity(), list);
            xListView.setAdapter(adapter);
        }else {
            adapter.notifyDataSetChanged();
        }
    }
    @Override
    public void onRefresh() {
        //判断网络状态,异步加载数据
        if (NetConnectionUtil.isNetConnectioned(getActivity())){
            getDataFromNet(data);
        }else{
            NetConnectionUtil.setNetConnectionWork(getActivity());
        }
    }

    @Override
    public void onLoadMore() {
        //判断网络状态,异步加载数据
        if (NetConnectionUtil.isNetConnectioned(getActivity())){
            getDataFromNet(data);
        }else{
            NetConnectionUtil.setNetConnectionWork(getActivity());
        }
    }
}

五. 自定义bean类

六. util工具类包:

1. 全局化配置ImageLoader类的 BaseApplication.java

package com.util;

import android.app.Application;
//全局初始化Application类
public class BaseApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();

        //配置imageLoader
        ImageLoaderUtil.init(this);
    }
}

2. 配置imageLoader的 ImageLoaderUtil.java

package com.util;

import android.content.Context;
import android.graphics.Bitmap;
import com.month.R;
import com.nostra13.universalimageloader.cache.disc.impl.UnlimitedDiscCache;
import com.nostra13.universalimageloader.cache.disc.naming.HashCodeFileNameGenerator;
import com.nostra13.universalimageloader.cache.memory.impl.LruMemoryCache;
import com.nostra13.universalimageloader.core.DisplayImageOptions;
import com.nostra13.universalimageloader.core.ImageLoader;
import com.nostra13.universalimageloader.core.ImageLoaderConfiguration;
import com.nostra13.universalimageloader.core.assist.ImageScaleType;
import com.nostra13.universalimageloader.core.assist.QueueProcessingType;
import com.nostra13.universalimageloader.core.decode.BaseImageDecoder;
import com.nostra13.universalimageloader.core.display.RoundedBitmapDisplayer;
import com.nostra13.universalimageloader.core.display.SimpleBitmapDisplayer;
import com.nostra13.universalimageloader.core.download.BaseImageDownloader;
import com.nostra13.universalimageloader.utils.StorageUtils;
import java.io.File;
public class ImageLoaderUtil {
    /**
     * 初始化imageLoader
     * @param context
     */
    public static void init(Context context) {
        //1.获取配置config对象
        File cacheDir = StorageUtils.getCacheDirectory(context);  //缓存文件夹路径

        ImageLoaderConfiguration config = new ImageLoaderConfiguration.Builder(context)

                .threadPoolSize(3) // default  线程池内加载的数量
                .threadPriority(Thread.NORM_PRIORITY - 2) // default 设置当前线程的优先级
                .tasksProcessingOrder(QueueProcessingType.FIFO) // default
                .denyCacheImageMultipleSizesInMemory()
                .memoryCache(new LruMemoryCache(2 * 1024 * 1024)) //可以通过自己的内存缓存实现
                .memoryCacheSize(2 * 1024 * 1024)  // 内存缓存的最大值
                .memoryCacheSizePercentage(13) // default
                .diskCache(new UnlimitedDiscCache(cacheDir)) // default 可以自定义缓存路径
                .diskCacheSize(50 * 1024 * 1024) // 50 Mb sd卡(本地)缓存的最大值
                .diskCacheFileCount(100)  // 可以缓存的文件数量
                // default为使用HASHCODE对UIL进行加密命名, 还可以用MD5(new Md5FileNameGenerator())加密
                .diskCacheFileNameGenerator(new HashCodeFileNameGenerator())
                .imageDownloader(new BaseImageDownloader(context)) // default
                .imageDecoder(new BaseImageDecoder(true)) // default
                .defaultDisplayImageOptions(DisplayImageOptions.createSimple()) // default
                .writeDebugLogs() // 打印debug log
                .build(); //开始构建


        //2.初始化配置...ImageLoader.getInstance()图片加载器的对象,单例模式
        ImageLoader.getInstance().init(config);
    }

    /**
     * imageLoader加载图片的默认选项
     * @return
     */
    public static DisplayImageOptions getDefaultOption(){

        DisplayImageOptions options = new DisplayImageOptions.Builder()
                .showImageOnLoading(R.drawable.app) // 设置图片下载期间显示的默认图片
                .showImageForEmptyUri(R.drawable.app) // 设置图片Uri为空或是错误的时候显示的图片
                .showImageOnFail(R.drawable.app) // 设置图片加载或解码过程中发生错误显示的图片
                .resetViewBeforeLoading(true)  // default 设置图片在加载前是否重置、复位
                .delayBeforeLoading(1000)  // 下载前的延迟时间
                .cacheInMemory(true) // default  设置下载的图片是否缓存在内存中
                .cacheOnDisk(true) // default  设置下载的图片是否缓存在SD卡中

                .considerExifParams(true) // default
                .imageScaleType(ImageScaleType.IN_SAMPLE_POWER_OF_2) // default 设置图片以如何的编码方式显示
                .bitmapConfig(Bitmap.Config.RGB_565) // default 设置图片的解码类型

                .displayer(new SimpleBitmapDisplayer()) // default  还可以设置圆角图片new RoundedBitmapDisplayer(20)

                .build();

        return options;
    }

    /**
     * imageLoader加载圆角图片....指定圆角的大小
     * @return
     */
    public static DisplayImageOptions getRoundedOption(int corner){

        DisplayImageOptions options = new DisplayImageOptions.Builder()
                .showImageOnLoading(R.drawable.app) // 设置图片下载期间显示的图片
                .showImageForEmptyUri(R.drawable.app) // 设置图片Uri为空或是错误的时候显示的图片
                .showImageOnFail(R.drawable.app) // 设置图片加载或解码过程中发生错误显示的图片
                .resetViewBeforeLoading(true)  // default 设置图片在加载前是否重置、复位
                .delayBeforeLoading(1000)  // 下载前的延迟时间
                .cacheInMemory(true) // default  设置下载的图片是否缓存在内存中
                .cacheOnDisk(true) // default  设置下载的图片是否缓存在SD卡中

                .considerExifParams(true) // default
                .imageScaleType(ImageScaleType.IN_SAMPLE_POWER_OF_2) // default 设置图片以如何的编码方式显示
                .bitmapConfig(Bitmap.Config.RGB_565) // default 设置图片的解码类型

                .displayer(new RoundedBitmapDisplayer(corner)) // default  还可以设置圆角图片new RoundedBitmapDisplayer(20)

                .build();

        return options;
    }

}

3. 判断网络连接的工具类  NetConnectionUtil.java

package com.util;
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.provider.Settings;
/**
 * 判断网络连接的工具类
 */
public class NetConnectionUtil {
    /**
     * 判断是否有网络连接的方法
     */
    public  static boolean isNetConnectioned(Context context){
        //1. 获取网络连接对象
        ConnectivityManager manager = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);

        //2. 获取NetworkInfo对象,获取网络连接的信息
        NetworkInfo info = manager.getActiveNetworkInfo();

        //3. 信息不为空时,就代表网络可用
        if (info != null){
            return info.isAvailable();
        }
        return false;
    }

    /**
     * 网络无连接时跳转页面,弹出对话框进行网络的设置
     */
    public static void setNetConnectionWork(final Context context){
        AlertDialog.Builder builder = new AlertDialog.Builder(context);
        builder.setTitle("网络加载错误!");
        builder.setMessage("网络连接不可用,是否设置网络?");
        builder.setPositiveButton("确定", new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialogInterface, int i) {
                //跳转到系统的设置网络的界面
                Intent intent = new Intent(Settings.ACTION_WIFI_SETTINGS);
                context.startActivity(intent);
            }
        });
        builder.setNegativeButton("取消",null);
        builder.show();
    }
}

4. 解析json数据的  StringUtil.java

package com.util;
import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;

public class StringUtil {

    public static String streamToString(InputStream inputStream, String charset) {
        try {
            InputStreamReader inputStreamReader = new InputStreamReader(inputStream,charset);

            BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
            String s = null;
            StringBuilder builder = new StringBuilder();
            while ((s = bufferedReader.readLine()) != null){
                builder.append(s);
            }

            bufferedReader.close();
            return builder.toString();

        } catch (Exception e) {
            e.printStackTrace();
        }

        return  null;
    }
}


七.  view 包下的 XListView实现类:

1. XListView.java

/**
 * @file XListView.java
 * @package me.maxwin.view
 * @create Mar 18, 2012 6:28:41 PM
 * @author Maxwin
 * @description An ListView support (a) Pull down to refresh, (b) Pull up to load more.
 *        Implement IXListViewListener, and see stopRefresh() / stopLoadMore().
 */
package view;

import androidthree_1509d.MyViewpager.R;
import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewTreeObserver.OnGlobalLayoutListener;
import android.view.animation.DecelerateInterpolator;
import android.widget.AbsListView;
import android.widget.AbsListView.OnScrollListener;
import android.widget.ListAdapter;
import android.widget.ListView;
import android.widget.RelativeLayout;
import android.widget.Scroller;
import android.widget.TextView;

public class XListView extends ListView implements OnScrollListener {

   private float mLastY = -1; // save event y
   private Scroller mScroller; // used for scroll back
   private OnScrollListener mScrollListener; // user's scroll listener

   // the interface to trigger refresh and load more.
   private IXListViewListener mListViewListener;

   // -- header view
   private XListViewHeader mHeaderView;
   // header view content, use it to calculate the Header's height. And hide it
   // when disable pull refresh.
   private RelativeLayout mHeaderViewContent;
   private TextView mHeaderTimeView;
   private int mHeaderViewHeight; // header view's height
   private boolean mEnablePullRefresh = true;
   private boolean mPullRefreshing = false; // is refreashing.

   // -- footer view
   private XListViewFooter mFooterView;
   private boolean mEnablePullLoad;
   private boolean mPullLoading;
   private boolean mIsFooterReady = false;
   
   // total list items, used to detect is at the bottom of listview.
   private int mTotalItemCount;

   // for mScroller, scroll back from header or footer.
   private int mScrollBack;
   private final static int SCROLLBACK_HEADER = 0;
   private final static int SCROLLBACK_FOOTER = 1;

   private final static int SCROLL_DURATION = 400; // scroll back duration
   private final static int PULL_LOAD_MORE_DELTA = 50; // when pull up >= 50px
                                          // at bottom, trigger
                                          // load more.
   private final static float OFFSET_RADIO = 1.8f; // support iOS like pull
                                       // feature.

   /**
    * @param context
    */
   public XListView(Context context) {
      super(context);
      initWithContext(context);
   }

   public XListView(Context context, AttributeSet attrs) {
      super(context, attrs);
      initWithContext(context);
   }

   public XListView(Context context, AttributeSet attrs, int defStyle) {
      super(context, attrs, defStyle);
      initWithContext(context);
   }

   private void initWithContext(Context context) {
      mScroller = new Scroller(context, new DecelerateInterpolator());
      // XListView need the scroll event, and it will dispatch the event to
      // user's listener (as a proxy).
      super.setOnScrollListener(this);

      // init header view
      mHeaderView = new XListViewHeader(context);
      mHeaderViewContent = (RelativeLayout) mHeaderView
            .findViewById(R.id.xlistview_header_content);
      mHeaderTimeView = (TextView) mHeaderView
            .findViewById(R.id.xlistview_header_time);
      addHeaderView(mHeaderView);

      // init footer view
      mFooterView = new XListViewFooter(context);

      // init header height
      mHeaderView.getViewTreeObserver().addOnGlobalLayoutListener(
            new OnGlobalLayoutListener() {
               @Override
               public void onGlobalLayout() {
                  mHeaderViewHeight = mHeaderViewContent.getHeight();
                  getViewTreeObserver()
                        .removeGlobalOnLayoutListener(this);
               }
            });
   }

   @Override
   public void setAdapter(ListAdapter adapter) {
      // make sure XListViewFooter is the last footer view, and only add once.
      if (mIsFooterReady == false) {
         mIsFooterReady = true;
         addFooterView(mFooterView);
      }
      super.setAdapter(adapter);
   }

   /**
    * enable or disable pull down refresh feature.
    * 
    * @param enable
    */
   public void setPullRefreshEnable(boolean enable) {
      mEnablePullRefresh = enable;
      if (!mEnablePullRefresh) { // disable, hide the content
         mHeaderViewContent.setVisibility(View.INVISIBLE);
      } else {
         mHeaderViewContent.setVisibility(View.VISIBLE);
      }
   }

   /**
    * enable or disable pull up load more feature.
    * 
    * @param enable
    */
   public void setPullLoadEnable(boolean enable) {
      mEnablePullLoad = enable;
      if (!mEnablePullLoad) {
         mFooterView.hide();
         mFooterView.setOnClickListener(null);
         //make sure "pull up" don't show a line in bottom when listview with one page 
         setFooterDividersEnabled(false);
      } else {
         mPullLoading = false;
         mFooterView.show();
         mFooterView.setState(XListViewFooter.STATE_NORMAL);
         //make sure "pull up" don't show a line in bottom when listview with one page  
         setFooterDividersEnabled(true);
         // both "pull up" and "click" will invoke load more.
         mFooterView.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
               startLoadMore();
            }
         });
      }
   }

   /**
    * stop refresh, reset header view.
    */
   public void stopRefresh() {
      if (mPullRefreshing == true) {
         mPullRefreshing = false;
         resetHeaderHeight();
      }
   }

   /**
    * stop load more, reset footer view.
    */
   public void stopLoadMore() {
      if (mPullLoading == true) {
         mPullLoading = false;
         mFooterView.setState(XListViewFooter.STATE_NORMAL);
      }
   }

   /**
    * set last refresh time
    * 
    * @param time
    */
   public void setRefreshTime(String time) {
      mHeaderTimeView.setText(time);
   }

   private void invokeOnScrolling() {
      if (mScrollListener instanceof OnXScrollListener) {
         OnXScrollListener l = (OnXScrollListener) mScrollListener;
         l.onXScrolling(this);
      }
   }

   private void updateHeaderHeight(float delta) {
      mHeaderView.setVisiableHeight((int) delta
            + mHeaderView.getVisiableHeight());
      if (mEnablePullRefresh && !mPullRefreshing) { // 未处于刷新状态,更新箭头
         if (mHeaderView.getVisiableHeight() > mHeaderViewHeight) {
            mHeaderView.setState(XListViewHeader.STATE_READY);
         } else {
            mHeaderView.setState(XListViewHeader.STATE_NORMAL);
         }
      }
      setSelection(0); // scroll to top each time
   }

   /**
    * reset header view's height.
    */
   private void resetHeaderHeight() {
      int height = mHeaderView.getVisiableHeight();
      if (height == 0) // not visible.
         return;
      // refreshing and header isn't shown fully. do nothing.
      if (mPullRefreshing && height <= mHeaderViewHeight) {
         return;
      }
      int finalHeight = 0; // default: scroll back to dismiss header.
      // is refreshing, just scroll back to show all the header.
      if (mPullRefreshing && height > mHeaderViewHeight) {
         finalHeight = mHeaderViewHeight;
      }
      mScrollBack = SCROLLBACK_HEADER;
      mScroller.startScroll(0, height, 0, finalHeight - height,
            SCROLL_DURATION);
      // trigger computeScroll
      invalidate();
   }

   private void updateFooterHeight(float delta) {
      int height = mFooterView.getBottomMargin() + (int) delta;
      if (mEnablePullLoad && !mPullLoading) {
         if (height > PULL_LOAD_MORE_DELTA) { // height enough to invoke load
                                       // more.
            mFooterView.setState(XListViewFooter.STATE_READY);
         } else {
            mFooterView.setState(XListViewFooter.STATE_NORMAL);
         }
      }
      mFooterView.setBottomMargin(height);

//    setSelection(mTotalItemCount - 1); // scroll to bottom
   }

   private void resetFooterHeight() {
      int bottomMargin = mFooterView.getBottomMargin();
      if (bottomMargin > 0) {
         mScrollBack = SCROLLBACK_FOOTER;
         mScroller.startScroll(0, bottomMargin, 0, -bottomMargin,
               SCROLL_DURATION);
         invalidate();
      }
   }

   private void startLoadMore() {
      mPullLoading = true;
      mFooterView.setState(XListViewFooter.STATE_LOADING);
      if (mListViewListener != null) {
         mListViewListener.onLoadMore();
      }
   }

   @Override
   public boolean onTouchEvent(MotionEvent ev) {
      if (mLastY == -1) {
         mLastY = ev.getRawY();
      }

      switch (ev.getAction()) {
      case MotionEvent.ACTION_DOWN:
         mLastY = ev.getRawY();
         break;
      case MotionEvent.ACTION_MOVE:
         final float deltaY = ev.getRawY() - mLastY;
         mLastY = ev.getRawY();
         if (getFirstVisiblePosition() == 0
               && (mHeaderView.getVisiableHeight() > 0 || deltaY > 0)) {
            // the first item is showing, header has shown or pull down.
            updateHeaderHeight(deltaY / OFFSET_RADIO);
            invokeOnScrolling();
         } else if (getLastVisiblePosition() == mTotalItemCount - 1
               && (mFooterView.getBottomMargin() > 0 || deltaY < 0)) {
            // last item, already pulled up or want to pull up.
            updateFooterHeight(-deltaY / OFFSET_RADIO);
         }
         break;
      default:
         mLastY = -1; // reset
         if (getFirstVisiblePosition() == 0) {
            // invoke refresh
            if (mEnablePullRefresh
                  && mHeaderView.getVisiableHeight() > mHeaderViewHeight) {
               mPullRefreshing = true;
               mHeaderView.setState(XListViewHeader.STATE_REFRESHING);
               if (mListViewListener != null) {
                  mListViewListener.onRefresh();
               }
            }
            resetHeaderHeight();
         } else if (getLastVisiblePosition() == mTotalItemCount - 1) {
            // invoke load more.
            if (mEnablePullLoad
                && mFooterView.getBottomMargin() > PULL_LOAD_MORE_DELTA
                && !mPullLoading) {
               startLoadMore();
            }
            resetFooterHeight();
         }
         break;
      }
      return super.onTouchEvent(ev);
   }

   @Override
   public void computeScroll() {
      if (mScroller.computeScrollOffset()) {
         if (mScrollBack == SCROLLBACK_HEADER) {
            mHeaderView.setVisiableHeight(mScroller.getCurrY());
         } else {
            mFooterView.setBottomMargin(mScroller.getCurrY());
         }
         postInvalidate();
         invokeOnScrolling();
      }
      super.computeScroll();
   }

   @Override
   public void setOnScrollListener(OnScrollListener l) {
      mScrollListener = l;
   }

   @Override
   public void onScrollStateChanged(AbsListView view, int scrollState) {
      if (mScrollListener != null) {
         mScrollListener.onScrollStateChanged(view, scrollState);
      }
   }

   @Override
   public void onScroll(AbsListView view, int firstVisibleItem,
         int visibleItemCount, int totalItemCount) {
      // send to user's listener
      mTotalItemCount = totalItemCount;
      if (mScrollListener != null) {
         mScrollListener.onScroll(view, firstVisibleItem, visibleItemCount,
               totalItemCount);
      }
   }

   public void setXListViewListener(IXListViewListener l) {
      mListViewListener = l;
   }

   /**
    * you can listen ListView.OnScrollListener or this one. it will invoke
    * onXScrolling when header/footer scroll back.
    */
   public interface OnXScrollListener extends OnScrollListener {
      public void onXScrolling(View view);
   }

   /**
    * implements this interface to get refresh/load more event.
    */
   public interface IXListViewListener {
      public void onRefresh();

      public void onLoadMore();
   }
}

2. XlistViewHeader.java

/**
 * @file XListViewHeader.java
 * @create Apr 18, 2012 5:22:27 PM
 * @author Maxwin
 * @description XListView's header
 */
package view;

import androidthree_1509d.MyViewpager.R;
import android.content.Context;
import android.util.AttributeSet;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.RotateAnimation;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.ProgressBar;
import android.widget.TextView;

public class XListViewHeader extends LinearLayout {
   private LinearLayout mContainer;
   private ImageView mArrowImageView;
   private ProgressBar mProgressBar;
   private TextView mHintTextView;
   private int mState = STATE_NORMAL;

   private Animation mRotateUpAnim;
   private Animation mRotateDownAnim;
   
   private final int ROTATE_ANIM_DURATION = 180;
   
   public final static int STATE_NORMAL = 0;
   public final static int STATE_READY = 1;
   public final static int STATE_REFRESHING = 2;

   public XListViewHeader(Context context) {
      super(context);
      initView(context);
   }

   /**
    * @param context
    * @param attrs
    */
   public XListViewHeader(Context context, AttributeSet attrs) {
      super(context, attrs);
      initView(context);
   }

   private void initView(Context context) {
      // 初始情况,设置下拉刷新view高度为0
      LayoutParams lp = new LayoutParams(
            LayoutParams.FILL_PARENT, 0);
      mContainer = (LinearLayout) LayoutInflater.from(context).inflate(
            R.layout.xlistview_header, null);
      addView(mContainer, lp);
      setGravity(Gravity.BOTTOM);

      mArrowImageView = (ImageView)findViewById(R.id.xlistview_header_arrow);
      mHintTextView = (TextView)findViewById(R.id.xlistview_header_hint_textview);
      mProgressBar = (ProgressBar)findViewById(R.id.xlistview_header_progressbar);
      
      mRotateUpAnim = new RotateAnimation(0.0f, -180.0f,
            Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF,
            0.5f);
      mRotateUpAnim.setDuration(ROTATE_ANIM_DURATION);
      mRotateUpAnim.setFillAfter(true);
      mRotateDownAnim = new RotateAnimation(-180.0f, 0.0f,
            Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF,
            0.5f);
      mRotateDownAnim.setDuration(ROTATE_ANIM_DURATION);
      mRotateDownAnim.setFillAfter(true);
   }

   public void setState(int state) {
      if (state == mState) return ;
      
      if (state == STATE_REFRESHING) {   // 显示进度
         mArrowImageView.clearAnimation();
         mArrowImageView.setVisibility(View.INVISIBLE);
         mProgressBar.setVisibility(View.VISIBLE);
      } else {   // 显示箭头图片
         mArrowImageView.setVisibility(View.VISIBLE);
         mProgressBar.setVisibility(View.INVISIBLE);
      }
      
      switch(state){
      case STATE_NORMAL:
         if (mState == STATE_READY) {
            mArrowImageView.startAnimation(mRotateDownAnim);
         }
         if (mState == STATE_REFRESHING) {
            mArrowImageView.clearAnimation();
         }
         mHintTextView.setText(R.string.xlistview_header_hint_normal);
         break;
      case STATE_READY:
         if (mState != STATE_READY) {
            mArrowImageView.clearAnimation();
            mArrowImageView.startAnimation(mRotateUpAnim);
            mHintTextView.setText(R.string.xlistview_header_hint_ready);
         }
         break;
      case STATE_REFRESHING:
         mHintTextView.setText(R.string.xlistview_header_hint_loading);
         break;
         default:
      }
      
      mState = state;
   }
   
   public void setVisiableHeight(int height) {
      if (height < 0)
         height = 0;
      LayoutParams lp = (LayoutParams) mContainer
            .getLayoutParams();
      lp.height = height;
      mContainer.setLayoutParams(lp);
   }

   public int getVisiableHeight() {
      return mContainer.getLayoutParams().height;
   }

}

3. XListViewFooter.java

/**
 * @file XFooterView.java
 * @create Mar 31, 2012 9:33:43 PM
 * @author Maxwin
 * @description XListView's footer
 */
package view;
import androidthree_1509d.MyViewpager.R;
import android.content.Context;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.LinearLayout;
import android.widget.TextView;

public class XListViewFooter extends LinearLayout {
   public final static int STATE_NORMAL = 0;
   public final static int STATE_READY = 1;
   public final static int STATE_LOADING = 2;

   private Context mContext;

   private View mContentView;
   private View mProgressBar;
   private TextView mHintView;
   
   public XListViewFooter(Context context) {
      super(context);
      initView(context);
   }
   
   public XListViewFooter(Context context, AttributeSet attrs) {
      super(context, attrs);
      initView(context);
   }
   
   public void setState(int state) {
      mHintView.setVisibility(View.INVISIBLE);
      mProgressBar.setVisibility(View.INVISIBLE);
      mHintView.setVisibility(View.INVISIBLE);
      if (state == STATE_READY) {
         mHintView.setVisibility(View.VISIBLE);
         mHintView.setText(R.string.xlistview_footer_hint_ready);
      } else if (state == STATE_LOADING) {
         mProgressBar.setVisibility(View.VISIBLE);
      } else {
         mHintView.setVisibility(View.VISIBLE);
         mHintView.setText(R.string.xlistview_footer_hint_normal);
      }
   }
   
   public void setBottomMargin(int height) {
      if (height < 0) return ;
      LayoutParams lp = (LayoutParams)mContentView.getLayoutParams();
      lp.bottomMargin = height;
      mContentView.setLayoutParams(lp);
   }
   
   public int getBottomMargin() {
      LayoutParams lp = (LayoutParams)mContentView.getLayoutParams();
      return lp.bottomMargin;
   }
   
   
   /**
    * normal status
    */
   public void normal() {
      mHintView.setVisibility(View.VISIBLE);
      mProgressBar.setVisibility(View.GONE);
   }
   
   
   /**
    * loading status 
    */
   public void loading() {
      mHintView.setVisibility(View.GONE);
      mProgressBar.setVisibility(View.VISIBLE);
   }
   
   /**
    * hide footer when disable pull load more
    */
   public void hide() {
      LayoutParams lp = (LayoutParams)mContentView.getLayoutParams();
      lp.height = 0;
      mContentView.setLayoutParams(lp);
   }
   
   /**
    * show footer
    */
   public void show() {
      LayoutParams lp = (LayoutParams)mContentView.getLayoutParams();
      lp.height = LayoutParams.WRAP_CONTENT;
      mContentView.setLayoutParams(lp);
   }
   
   private void initView(Context context) {
      mContext = context;
      LinearLayout moreView = (LinearLayout)LayoutInflater.from(mContext).inflate(R.layout.xlistview_footer, null);
      addView(moreView);
      moreView.setLayoutParams(new LayoutParams(LayoutParams.FILL_PARENT, LayoutParams.WRAP_CONTENT));
      
      mContentView = moreView.findViewById(R.id.xlistview_footer_content);
      mProgressBar = moreView.findViewById(R.id.xlistview_footer_progressbar);
      mHintView = (TextView)moreView.findViewById(R.id.xlistview_footer_hint_textview);
   }
   
   
}

八. 自定义的布局文件:

1. activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <com.Horizaontal.MyIndicator
        android:background="@android:color/holo_blue_bright"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:id="@+id/indicator"></com.Horizaontal.MyIndicator>

    <android.support.v4.view.ViewPager
        android:id="@+id/viewPager"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
</LinearLayout>

2. 显示fragment的news_layout.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">

    <view.XListView
        android:scrollbars="none"
        android:id="@+id/XListView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"></view.XListView>
</LinearLayout>

3. 多条目展示数据的4种布局

(1)item_layout01.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="horizontal"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <LinearLayout
        android:layout_weight="2"
        android:layout_width="0dp"
        android:orientation="vertical"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true">

            <TextView
                android:textSize="18sp"
                android:id="@+id/title"
                android:textStyle="bold"
                android:layout_marginLeft="10dp"
                android:layout_marginTop="18dp"
                android:layout_width="match_parent"
                android:layout_height="wrap_content" />

            <TextView
                android:textSize="13sp"
                android:id="@+id/author_name"
                android:layout_marginLeft="10dp"
                android:layout_marginTop="10dp"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content" />
    </LinearLayout>

    <ImageView
        android:id="@+id/image"
        android:layout_weight="1"
        android:layout_width="0dp"
        android:layout_height="100dp"
        android:layout_marginRight="5dp"
        android:layout_gravity="center_vertical" />
</LinearLayout>

(2)item_layout02.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:textSize="18sp"
        android:id="@+id/title"
        android:textStyle="bold"
        android:layout_marginLeft="10dp"
        android:layout_marginTop="20dp"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <LinearLayout
        android:orientation="horizontal"
        android:layout_marginTop="18dp"
        android:layout_marginLeft="10dp"
        android:layout_marginRight="10dp"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" >

        <ImageView
            android:layout_weight="1"
            android:layout_width="0dp"
            android:id="@+id/image001"
            android:layout_height="120dp"   />

        <ImageView
            android:layout_weight="1"
            android:layout_width="0dp"
            android:id="@+id/image002"
            android:layout_height="120dp"
            android:layout_marginLeft="3dp"/>

        <ImageView
            android:layout_weight="1"
            android:layout_width="0dp"
            android:layout_height="120dp"
            android:layout_marginLeft="3dp"/>

    </LinearLayout>

</LinearLayout>

(3)item_layout03.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:textSize="18sp"
        android:id="@+id/title"
        android:textStyle="bold"
        android:layout_marginLeft="10dp"
        android:layout_marginTop="20dp"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <LinearLayout
        android:orientation="horizontal"
        android:layout_marginTop="18dp"
        android:layout_marginLeft="10dp"
        android:layout_marginRight="10dp"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" >

        <ImageView
            android:layout_weight="1"
            android:id="@+id/image01"
            android:layout_width="0dp"
            android:layout_height="120dp"   />

        <ImageView
            android:layout_weight="1"
            android:id="@+id/image02"
            android:layout_width="0dp"
            android:layout_height="120dp"
            android:layout_marginLeft="3dp"/>

        <ImageView
            android:layout_weight="1"
            android:id="@+id/image03"
            android:layout_width="0dp"
            android:layout_height="120dp"
            android:layout_marginLeft="3dp"/>

    </LinearLayout>

</LinearLayout>

(4)title _layout.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">

    <TextView
        android:id="@+id/title"
        android:textSize="18sp"
        android:layout_height="60dp"
        android:gravity="center_vertical"
        android:layout_marginLeft="10dp"
        android:layout_width="wrap_content"    />
</LinearLayout>

4. 展示xlistview布局的2个xml
(1)xlistview_header.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:gravity="bottom" >

    <RelativeLayout
        android:id="@+id/xlistview_header_content"
        android:layout_width="fill_parent"
        android:layout_height="60dp" >

        <LinearLayout
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:gravity="center"
            android:orientation="vertical" android:id="@+id/xlistview_header_text">

            <TextView
                android:id="@+id/xlistview_header_hint_textview"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="@string/xlistview_header_hint_normal" />

            <LinearLayout
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_marginTop="3dp" >

                <TextView
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:text="@string/xlistview_header_last_time"
                    android:textSize="12sp" />

                <TextView
                    android:id="@+id/xlistview_header_time"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:textSize="12sp" />
            </LinearLayout>
        </LinearLayout>

        <ImageView
            android:id="@+id/xlistview_header_arrow"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_alignLeft="@id/xlistview_header_text"
            android:layout_centerVertical="true"
            android:layout_marginLeft="-35dp"
            android:src="@drawable/xlistview_arrow" />

        <ProgressBar
            android:id="@+id/xlistview_header_progressbar"
            android:layout_width="30dp"
            android:layout_height="30dp"
            android:layout_alignLeft="@id/xlistview_header_text"
            android:layout_centerVertical="true"
            android:layout_marginLeft="-40dp"
            android:visibility="invisible" />
    </RelativeLayout>

</LinearLayout>

(2)xlistview_footer.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content" >

    <RelativeLayout
        android:id="@+id/xlistview_footer_content"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:padding="10dp" >

        <ProgressBar
            android:id="@+id/xlistview_footer_progressbar"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:visibility="invisible" />

        <TextView
            android:id="@+id/xlistview_footer_hint_textview"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:text="@string/xlistview_footer_hint_normal" />
    </RelativeLayout>

</LinearLayout>

5. values文件夹下的布局文件strings.xml添加以下代码:

<string name="xlistview_header_hint_normal">下拉刷新</string>
<string name="xlistview_header_hint_ready">松开刷新数据</string>
<string name="xlistview_header_hint_loading">正在加载...</string>
<string name="xlistview_header_last_time">上次更新时间:</string>
<string name="xlistview_footer_hint_normal">查看更多</string>
<string name="xlistview_footer_hint_ready">松开载入更多</string>

十. AndroidManifest.xml中

 1. 加入请求网络数据 ,判断网络状态的权限;

 2.声明imageloader的name属性。


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值