android高级ui10-recycleview回收复用自定义+界面拖动效果

学习笔记,整理中;

面:

思路:

复用:从集合中去取

复用:从集合中去获取

入口:滑动 Move 事件 --> scrollByInternal --> scrollStep --> mLayout.scrollVerticallyBy

--> scrollBy  --> fill --> layoutChunk  布局块--> layoutState.next 布局状态--> addView(view);

layoutState.next --> getViewForPosition --> tryGetViewHolderForPositionByDeadline 截止日期 -->

怎么从集合中去获取:tryGetViewHolderForPositionByDeadline,分几种情况去获取ViewHolder

1. getChangedScrapViewForPosition -- mChangeScrap 与动画相关

2. getScrapOrHiddenOrCachedHolderForPosition  -- mAttachedScrap附属废料 、mCachedViews 缓存

3. getScrapOrCachedViewForId  -- mAttachedScrap 、mCachedViews (ViewType,itemid)

4. mViewCacheExtension.getViewForPositionAndType -- 自定义缓存 -- (使用情况:局部刷新??)

5. getRecycledViewPool().getRecycledView -- 从缓冲池里面获取

RecycledViewPool -- 缓存池

ViewHolder -- 包装View的 --- ItemView

当没有缓存的时候??--- mAdapter.createViewHolder --》 onCreateViewHolder

多级缓存的目的 -- 为了性能

创建ViewHolder 后 绑定: tryBindViewHolderByDeadline--》 mAdapter.bindViewHolder--》onBindViewHolder

回收(缓存)机制:看这一个情况--- ViewHolder

LinearLayoutManager.onLayoutChildren --> detachAndScrapAttachedViews 分离--> scrapOrRecycleView

--> 1.recycler.recycleViewHolderInternal(viewHolder); -- 处理 CacheView 、RecyclerViewPool 的缓存

--> 1.ViewHodler改变 不会进来 -- 先判断mCachedViews的大小

--> mCachedViews.size 大于默认大小  --- recycleCachedViewAt

--- >addViewHolderToRecycledViewPool --- 缓存池里面的数据都是从mCachedViews里面出来的

--> 2.addViewHolderToRecycledViewPool --> getRecycledViewPool().putRecycledView(holder);

--> scrap.resetInternal();  ViewHolder 清空

--> 2.recycler.scrapView(view);

mCachedViews当前的大小 如果 大于等于mViewCacheMax(默认的CachedViews的大小)

ViewType --

缓存池  里面保存  只是 ViewHolder 类型 没有数据

去查找缓存和复用的一种情况

入口:复用:RecyclerView.onLayout --> dispatchLayout 派遣--》 dispatchLayoutStep2 --》 onLayoutChildren --》 fill

缓存:fill -->recycleByLayoutState-->recycleViewsFromStart --> recycleChildren

--> removeAndRecycleViewAt --> recycler.recycleView

--> recycler.recycleViewHolderInternal(viewHolder); -- 处理 CacheView 、RecyclerViewPool 的缓存

要学会copy系统的代码 --- 怎么优化

layoutChunk --> layoutDecoratedWithMargins

notifyDataSetChanged-->mObservable.notifyChanged

--> (RecyclerViewDataObserver)mObservers.get(i).onChanged --> requestLayout


mChangeScrap与 mAttachedScrap

用来缓存还在屏幕内的 ViewHolder

2.mCachedViews

用来缓存移除屏幕之外的 ViewHolder

3.mViewCacheExtension

开发给用户的自定义扩展缓存,需要用户自己管理 View 的创建和缓存

4.RecycledViewPool

ViewHolder 缓存池


1

—————————-

自定义layoutmanage 继承recycleview.layout 可以实现布局改变 在界面上拖动或者拖拽,settranslationy拖动 setscrolx缩放

SlideCallback extends ItemTouchHelper.SimpleCallback

SlideCardLayoutManagerextends RecyclerView.LayoutManager

------------------------------------------------------------------------

public class MainActivityextends AppCompatActivity {

private RecyclerViewrv;

    private UniversalAdapteradapter;

    private ListmDatas;

    @Override

    protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);

        rv = findViewById(R.id.rv);

        rv.setLayoutManager(new SlideCardLayoutManager());

        mDatas = SlideCardBean.initDatas();

        adapter =new UniversalAdapter(this, mDatas, R.layout.item_swipe_card) {

@Override

            public void convert(ViewHolder viewHolder, SlideCardBean slideCardBean) {

viewHolder.setText(R.id.tvName, slideCardBean.getName());

                viewHolder.setText(R.id.tvPrecent, slideCardBean.getPostition() +"/" +mDatas.size());

                Glide.with(MainActivity.this)

.load(slideCardBean.getUrl())

.into((ImageView) viewHolder.getView(R.id.iv));

            }

};

        rv.setAdapter(adapter);

        // 初始化数据

        CardConfig.initConfig(this);

        SlideCallback slideCallback =new SlideCallback(rv, adapter, mDatas);

        ItemTouchHelper itemTouchHelper =new ItemTouchHelper(slideCallback);

        itemTouchHelper.attachToRecyclerView(rv);

    }

}

------------------------------------------------------------------------

public class SlideCardLayoutManagerextends RecyclerView.LayoutManager {

@Override

    public RecyclerView.LayoutParamsgenerateDefaultLayoutParams() {

return new RecyclerView.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT,

                ViewGroup.LayoutParams.WRAP_CONTENT);

    }

// 布局

    @Override

    public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {

// ViewHodler回收复用

        detachAndScrapAttachedViews(recycler);

        int bottomPosition;

        int itemCount = getItemCount();

        if (itemCount < CardConfig.MAX_SHOW_COUNT) {

bottomPosition =0;

        }else {

// 布局了四张卡片 --- 4,5,6,7

            bottomPosition = itemCount - CardConfig.MAX_SHOW_COUNT;

        }

for (int i = bottomPosition; i < itemCount; i++) {

// 复用

            View view = recycler.getViewForPosition(i);

            addView(view);

            measureChildWithMargins(view, 0, 0);

            int widthSpace = getWidth() - getDecoratedMeasuredWidth(view);

            int heightSpace = getHeight() - getDecoratedMeasuredHeight(view);

            // 布局 ---draw -- onDraw ,onDrawOver, onLayout

            layoutDecoratedWithMargins(view, widthSpace /2, heightSpace /2,

                    widthSpace /2 + getDecoratedMeasuredWidth(view),

                    heightSpace /2 + getDecoratedMeasuredHeight(view));

            int level = itemCount - i -1;

            if (level >0) {

if (level < CardConfig.MAX_SHOW_COUNT -1) {

view.setTranslationY(CardConfig.TRANS_Y_GAP * level);

                    view.setScaleX(1 - CardConfig.SCALE_GAP * level);

                    view.setScaleY(1 - CardConfig.SCALE_GAP * level);

                }else {

// 最下面的那个view 与前一个view 布局一样

                    view.setTranslationY(CardConfig.TRANS_Y_GAP * (level -1));

                    view.setScaleX(1 - CardConfig.SCALE_GAP * (level -1));

                    view.setScaleY(1 - CardConfig.SCALE_GAP * (level -1));

                }

}

}

}

}

------------------------------------------------------------------------

public class SlideCallbackextends ItemTouchHelper.SimpleCallback {

private RecyclerViewmRv;

    private UniversalAdapteradapter;

    private ListmDatas;

    public SlideCallback(RecyclerView mRv,

                        UniversalAdapter adapter, List mDatas) {

super(0, 15);

        this.mRv = mRv;

        this.adapter = adapter;

        this.mDatas = mDatas;

    }

// drag  拖拽

    @Override

    public boolean onMove(@NonNull RecyclerView recyclerView, @NonNull RecyclerView.ViewHolder viewHolder, @NonNull RecyclerView.ViewHolder target) {

return false;

    }

// 滑动

    @Override

    public void onSwiped(@NonNull RecyclerView.ViewHolder viewHolder, int direction) {

SlideCardBean remove =mDatas.remove(viewHolder.getLayoutPosition());

        mDatas.add(0, remove);

        adapter.notifyDataSetChanged();// onMeasure, onlayout

    }

// onDra

    @Override

    public void onChildDraw(@NonNull Canvas c, @NonNull RecyclerView recyclerView, @NonNull RecyclerView.ViewHolder viewHolder, float dX, float dY, int actionState, boolean isCurrentlyActive) {

super.onChildDraw(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive);

        double maxDistance = recyclerView.getWidth() *0.5f;

        double distance = Math.sqrt(dX * dX + dY * dY);

        double fraction = distance / maxDistance;

        if (fraction >1) {

fraction =1;

        }

// 显示的个数  4个

        int itemCount = recyclerView.getChildCount();

        for (int i =0; i < itemCount; i++) {

View view = recyclerView.getChildAt(i);

            int level = itemCount - i -1;

            if (level >0) {

if (level < CardConfig.MAX_SHOW_COUNT -1) {

view.setTranslationY((float) (CardConfig.TRANS_Y_GAP * level - fraction * CardConfig.TRANS_Y_GAP));

                    view.setScaleX((float) (1 - CardConfig.SCALE_GAP * level + fraction * CardConfig.SCALE_GAP));

                    view.setScaleY((float) (1 - CardConfig.SCALE_GAP * level + fraction * CardConfig.SCALE_GAP));

                }

}

}

}

@Override

    public long getAnimationDuration(@NonNull RecyclerView recyclerView, int animationType, float animateDx, float animateDy) {

return 1500;

    }

}

------------------------------------------------------------------------

------------------------------------------------------------------------

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容