跳至主要内容

[Widget]RecyclePagerAdapter-实现ViewPager复用回收的PagerAdapter

RecyclePagerAdapter-实现ViewPager复用回收的PagerAdapter

icon

实现ViewPager页卡View复用回收的PagerAdapter,只要是页面构造一样,则可以使用复用回收机制,同时也支持设置不同的viewType来实现多种样式的页卡回收复用,套用RecycleView的Adapter实现机制。

要求

  • com.android.support:support-v4
  • minSdkVersion 9
  • 保持跟其他官方支持库版本一致(如:com.android.support:appcompat-v7)

链接

引用

dependencies {
    ⋯
    compile 'am.project:supportplus:24.2.1.1'
    ⋯
}

使用

整体实现其实不难,使用过RecycleView的话,就可以轻车熟路,跟其实现方案一模一样。 实现自己的PagerViewHolder,个人习惯在实例化时进行布局inflate,这样打开PagerViewHolder便可以直接找到使用的布局文件:
public class MyPagerViewHolder extends RecyclePagerAdapter.PagerViewHolder {

    public MyPagerViewHolder(ViewGroup parent) {
        super(LayoutInflater.from(parent.getContext())
                .inflate(R.layout.item_recyclepager_page, parent, false));
    }
    //应用到页面上的数据
    public void setData(String data) {
        ((TextView) itemView).setText(data);
    }
}
实现自己的RecyclePagerAdapter:
public class MyRecyclePagerAdapter extends RecyclePagerAdapter<MyPagerViewHolder> {

    private int itemCount = 5;
    @Override
    public int getItemCount() {
        return itemCount;
    }

    @Override
    public MyPagerViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        //一般同viewType的Holder创建不会超过四个
        return new MyPagerViewHolder(parent);
    }

    @Override
    public void onBindViewHolder(MyPagerViewHolder holder, int position) {
        //处理不同页面的不同数据
        holder.setData(String.format(Locale.getDefault(),"第%d页", position + 1));
    }

    public void add() {
        itemCount++;
        notifyDataSetChanged();
    }

    public void remove() {
        itemCount--;
        itemCount = itemCount < 0 ? 0 : itemCount;
        notifyDataSetChanged();
    }

    @Override
    public int getItemViewType(int position) {
        //设置不同类型的页面。
        return 0;
    }

    @Override
    public void onViewRecycled(VH holder) {
        //当ViewPager执行destroyItem时,会回收Holder,此时会调用该方法,你可以重写该方法实现你要的效果
    }
}

原理

ViewPager最多构造四个相同类型的页面,但是现实时最多需要当前页面及左右两个页面,第四个页面就可以存起来复用。 Adapter使用一个ArrayList<VH>来存放所有的Holder;再用一个SparseArray<ArrayList<VH>>来根据viewType存放在destroyItem时候被回收的不同类型的Holder集合,在instantiateItem时候优先从其内部获取,在没有时再重新创建。

注意

  • notifyDataSetChanged()方法会将所有的未被回收的Holder重新onBindViewHolder一遍,并不是每一个ViewPager的页面都会刷新,但当前显示的绝对会刷新;
  • notifyItemChanged(int position)用于刷新指定的页面坐标的Holder,只有在这个页面处于激活状态时,其才会被刷新。

评论

此博客中的热门博文

[Widget]ShapeImageView-图形裁剪ImageView

ShapeImageView 图形裁剪ImageView,API 21 及以上 使用 setOutlineProvider 方式实现,支持动态图;以下使用 BitmapShader 方式实现。 支持固定高宽缩放比缩放,支持前景Drawable,支持ImageView的所有ScaleType,且API 21具备更高性能。 预览 要求 minSdkVersion 4 链接 Github Bintray 使用 基本布局 < am .widget.shapeimageview.ShapeImageView android : id = " @+id/siv_image_c " android : layout_width = " match_parent " android : layout_height = " wrap_content " android : layout_margin = " 5dp " android : layout_weight = " 1 " android : clickable = " true " android : scaleType = " centerCrop " android : src = " @drawable/bg_welcome " app : sivBorderColor = " @color/colorAccent " app : sivBorderWidth = " 2dp " app : sivForeground = " @drawable/bg_common_press_dark " app : sivHeightScale = " 1 " app : sivScaleTarget = " height " app : sivShape = &q

[Widget]StateFrameLayout-状态帧布局

StateFrameLayout icon 一般网络交互的状态提示及处理大多数情况下考虑使用Dialog,在一切状态处理理想状态下时,使用Dialog进行交互是可行的。但稍微一不注意,使用Dialog则会出现一系列隐藏的Bug。为节省用户时间怎加体验感觉,数据的载入可以在onCreate时候就进行,甚至可以在Activity构造函数里面启动网络请求,因为Activity还没有建立窗口(onAttachedToWindow),而Dialog必须附着在Activity的Window上,显然这时候不能弹出Dialog;网络交互并非即时,也就是在交互过程中用户可能进行任何操作,多数情况下,应用并不允许用户中断网络交互,而将Dialog设置为不可取消的话,用户体验是很差的,因为你同时阻止了用户退出当前Activity的操作,若用户仅仅是误点了进来,那么必须等待交互结束才能退出,而如果不讲Dialog设置为不可取消的话,那么用户进行了取消操作,但实际是并没有取消,这又会让用户很困惑,如果交互是更新当前页面的数据,当用户取消以后就可以进行旧数据操作,但其实这时候数据已过时,操作是不应该的;当网络交互已完成时,若交互结果需要告知用户时,此时又得注意Activity的状态,也许Activity已经关闭了Window(用户进行了返回操作,Activity在销毁;或者用户点按了Home键,设备内存不够,Activity在进行保存并关闭Window)。操控好Window,则使用Dialog并无任何问题,但是这就会怎加代码复杂度。其实我们的目的就是告知用户在进行网络请求,阻止用户对未载入或旧页面进行操作,网络交互结束后有必要时告知用户;使用StateFrameLayout则能轻松达到效果。 状态帧布局,通常用于网络请求的四种状态,普通、载入、错误、空白。支持Drawable或者View来展示,也可以混搭。 预览 screenshots 要求 minSdkVersion 4 链接 Github Bintray 使用 基本布局 < am .widget.stateframelayout.StateFrameLayout xmlns : app = " http://schemas.androi

[Skill]URLConnection从HTTP重定向到HTTPS

URLConnection从HTTP重定向到HTTPS   也不知什么原因,公司项目的服务端一直在吸引着大波攻击,于是服务端的同学打算把所有HTTP的请求都换为HTTPS,他们决定兼容旧版本于是就将之前的所有HTTP请求全部重定向到另一个HTTPS请求。 项目请求框架搭建初期,考虑到应用也不会使用太复杂的请求模式,于是就简单使用URLConnection完成服务端交互。服务端一修改,全部请求都失败了。虽然URLConnection有是否遵循重定向开关(setInstanceFollowRedirects),其默认就是开启的,即便你再强制其打开,也是没有用,问题依旧。找了大量资料,其实问题的关键点不是重定向而是从HTTP重定向到HTTPS,关键点就在URLConnection的两个子类上。 HttpURLConnection与HttpsURLConnection   HttpURLConnection为URLConnection的子类,而HttpsURLConnection为HttpURLConnection的子类,在HttpURLConnection基础上对HTTPS进行支持。 URLConnection通常使用URL的openConnection()方法获得,而URL是根据其是否为Https开头来打开一个HttpURLConnection还是HttpsURLConnection。 而当URLConnection进行connect()时,遇到了重定向,如果打开了遵循重定向,那么其会获取重定向的地址,然后尝试连接这个地址。值得注意的是,这时候并不是使用新的链接地址重新openConnection()一个URLConnection,而是直接尝试连接这个重定向的地址,否则也就不存在以上的Bug了。 于是理论上分析,HTTP重定向到HTTP是不存在问题的,HTTPS重定向到HTTPS也是不存在问题的,而HTTP与HTTPS之间的重定向,那么就很可能会有问题了。HTTP重定向到HTTPS,URLConnection会将重定向的HTTPS以HTTP方式继续提交,那么服务端肯定是认为你是错误的提交方式;同理,HTTPS重定向HTTP也一样。 问题解决 使用URLConnection抓取到重定向,就使用重定向的地址重新人为openConnection()一个新