AnimatedLinearLayout:带删除动画的LinearLayout

本文介绍一个自定义ViewGroup:AnimatedLinearLayout,在普通的LinearLayout基础上,它实现了删除某一项后,后面的项依次前移或上移的动画效果,效果如下:

AnimatedLinearLayout效果

GitHub下载地址

代码如下:
在onMeasure函数中加入init函数,该函数负责记录AnimatedLinearLayout下直属子view列表,并给每个子view添加点击事件,实现该子view点击消失,同时后面的view前移的效果。注意,这里由于AnimatedLinearLayout可能会使用wrap_content属性值,会导致onMeasure方法执行多次,所以我们加入了一个标志位isInited,只在第一次执行onMeasure的时候进行初始化操作。

public class AnimatedLinearLayout extends LinearLayout {
    ...
    private boolean isInited = false;
    ...
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {    
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);    
        if (!isInited) {        
            init();        
            isInited = true;    
        }
    }
    ...
}

下面是init函数的实现,主要是两个操作:将子view添加到directChildViewList,给子view添加点击事件。点击事件又分三步,执行给AnimatedLinearLayout设置的listener,隐藏被点击的子view,将排在该view之后的view依次往前移动一个位置。这里移动的动画使用ObjectAnimator实现,Android动画技术可以参考这篇文章。首先记录需要移动的view当前的偏移量curTranslationX,然后向左或向上移动一个距离。需要注意的是,getTranslationX获取的是该view相对最初位置在X轴方向移动的距离,而不是该view距离屏幕左上角远点的X轴距离,所以AnimatedLinearLayout里面所有子view初始的偏移量都是0,移动的次数越多,偏移量越大。

public class AnimatedLinearLayout extends LinearLayout {    
    private List<View> directChildViewList = new ArrayList<>();    
    private OnClickListener directChildOnClickListener = null;
    ...
    private void init() {    
        // 获取直属子view数量    
        int directChildCnt = getChildCount();    
        for (int i = 0; i < directChildCnt; i++) {        
            // 将直属子view添加到list中        
            final View directChildView = getChildAt(i);        
            directChildViewList.add(directChildView);        
            // 给直属子view添加点击响应        
            directChildView.setOnClickListener(new OnClickListener() {            
                @Override            
                public void onClick(View v) {                
                // 如果设置了子view的点击事件,先执行点击事件                
                if (null != directChildOnClickListener) {                    
                    directChildOnClickListener.onClick(v);                
                }                

                // 然后隐藏该view                
                directChildView.setVisibility(INVISIBLE);                
                // 并将排在该view之后的view依次往前移动一个位置,移动的距离为点击消失的view的宽度(或高度)               
                MarginLayoutParams  layoutParams = (MarginLayoutParams ) directChildView.getLayoutParams();                
                int currentPos = directChildViewList.indexOf(directChildView);                
                for (int j = currentPos + 1; j < directChildViewList.size(); j++) {                    
                    View viewToMove = directChildViewList.get(j);                    
                    if (getOrientation() == HORIZONTAL) {                        
                        float moveLeftVal = directChildView.getWidth() + layoutParams.leftMargin + layoutParams.rightMargin;                        
                        float curTranslationX = viewToMove.getTranslationX();                                                
                        ObjectAnimator moveLeft = ObjectAnimator.ofFloat(viewToMove, "translationX", curTranslationX, curTranslationX - moveLeftVal);                        
                        moveLeft.setDuration(1000);                        
                        moveLeft.start();                    
                    } else {                        
                        float moveTopVal = directChildView.getHeight() + layoutParams.topMargin + layoutParams.bottomMargin;                        
                        float curTranslationY = viewToMove.getTranslationY();                        
                        ObjectAnimator moveTop = ObjectAnimator.ofFloat(viewToMove, "translationY", curTranslationY, curTranslationY - moveTopVal);                        
                        moveTop.setDuration(1000);                        
                        moveTop.start();                    
                    }                
                  }            
                }       
            });    
        }
    }

    // 设置直属子view点击事件
    public void setDirectChildOnClickListener(OnClickListener listener) {    
        directChildOnClickListener = listener;
    }
}

AnimatedLinearLayout的使用:
布局文件

<HorizontalScrollView    
    android:layout_width="match_parent"    
    android:layout_height="40dp"    
    android:scrollbars="none">    

    <com.magic.wdl.animatedlinearlayout.AnimatedLinearLayout        
        android:id="@+id/animated_layout"        
        android:layout_width="wrap_content"        
        android:layout_height="wrap_content"        
        android:orientation="horizontal"        
        android:layout_gravity="center_vertical">        

        <TextView            
            android:layout_width="wrap_content"            
            android:layout_height="wrap_content"            
            android:text="远上寒山石径斜"            
            android:textSize="15sp"            
            android:layout_marginLeft="20dp" />        

        <TextView            
            android:layout_width="wrap_content"            
            android:layout_height="wrap_content"            
            android:text="白云深处有人家"            
            android:textSize="15sp"            
            android:layout_marginLeft="20dp"/>        
    </com.magic.wdl.animatedlinearlayout.AnimatedLinearLayout>
</HorizontalScrollView>

设置AnimatedLinearLayout子view点击事件:

public class MainActivity extends AppCompatActivity {
    ...
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        ...
        mAnimatedLinearLayoutHorizental.setDirectChildOnClickListener(getOnClickListener());
        ...
    }

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

推荐阅读更多精彩内容