• android View 关于transient


    今天来研究一下 ListView 的删除动画

    由于 ListView 卷动时会把画面上的 item 重用以显示不同数据

    这样会导致我们可能会删除到非正确的 item

    或是出现显示上的问题(该 item 显示的数据已经不同但是动画却还在它上面跑)

    要解决这个问题就要使用 View 类别在 API 16 加入的新 method


    public void setHasTransientState (boolean hasTransientState)

    当设为 true 的时候,就告诉系统这个 View 应该尽可能的被保留,直到setHasTransientState(false)被呼叫

    要注意这是可以重复设置的,也就是说每一个setHasTransientState(true)都要搭配一个setHasTransientState(false)才能回复

    //Listview 的 OnItemClickListener 的内容
    //本范例点了 item 后会淡出并删除该 item
    public void onItemClick(AdapterView
     
      parent, final View view, int position, long id) {
        final String item = (String) parent.getItemAtPosition(position);
        ObjectAnimator anim = ObjectAnimator.ofFloat(view, View.ALPHA, 0);
        anim.setDuration(500);
        view.setHasTransientState(true); //设为 true 宣告 item 要被追踪
        anim.addListener(new AnimatorListenerAdapter() {
            @Override
            public void onAnimationEnd(Animator animation) {
                myListview.remove(item);
                adapter.notifyDataSetChanged(); //重新整理 listview
                view.setAlpha(1);
                view.setHasTransientState(false); //完成后设定回 false
            }
        });
        anim.start();
    }
    

    另一种方法是用 ViewPropertyAnimator 来解决

    public void onItemClick(AdapterView
     
      parent, final View view, int position, long id) {
        final String item = (String) parent.getItemAtPosition(position);
        view.animate()
            .setDuration(500)
            .alpha(0)
            .withEndAction(new Runnable() {
                @Override
                public void run() {
                    myListview.remove(item);
                    adapter.notifyDataSetChanged();
                    view.setAlpha(1);
                }
            }
        );
    }
    

    但以上两种方式最大的问题就是 API 16 以后才能使用(用到了 withEndAction 跟 setHasTransientState)

    (该死的 Android Fragmentation…)

    让我们看看另一个例子

    在 DevBytes: Animating ListView Deletion: Now on Gingerbread! 里

    他用了一个继承自 ArrayAdapter 的自定义 adapter

    override 掉 public boolean hasStableIds() 让它总是回传 true ,用来取代 setHasTransientState(true)

    //用来取代 setHasTransientState(true)
    @Override
    public boolean hasStableIds() {
        return true; //总是回传 true,用来取代 setHasTransientState(true)
    }
    

    而 withEndAction 的部分则用 setAnimationListener 取代

    //API 11 之前的方法
    TranslateAnimation translator = new TranslateAnimation(startX, endX, startY, endY);
    translator.setDuration(150);
    view.startAnimation(translator);
    view.getAnimation().setAnimationListener(new AnimationListenerAdapter() {
        @Override
        public void onAnimationEnd(Animation animation) {
            new Runnable() {
                @Override
                public void run() {
                    //...(略)
                }
            }
        }
    });
    

    这样可以让 Gingerbread(API 10) 或更早的 Android 版本兼容

    如果是 4.0 (API 14)之后的版本

    还是用 view.animate() 跟 ObjectAnimator 来做

    //API 小于 16 但大于等于 14 可以这样做
    view.animate().setDuration(150);
    ObjectAnimator anim = ObjectAnimator.ofFloat(view, View.TRANSLATION_X, startX, endX);
    anim.setDuration(150);
    anim.start();
    setAnimatorEndAction(anim, new Runnable() {
        @Override
        public void run() {
            //...(略)
        }
    });
  • 相关阅读:
    查询是哪个进程占用了特定端口
    (面向c#开发人员) 编写javascript的好习惯一 false 值
    onerror 事件 如何使用 onerror 事件捕获网页中的错误。(chrome、opera、safari 浏览器不支持)
    查找url里面的flag元素判断操作
    jquery 弹出窗
    向上滚动
    kissy helpcenter
    kissyAPI
    IE6 动态创建 iframe 无法显示的 bug 芒果
    图片轮换动画仿GIF
  • 原文地址:https://www.cnblogs.com/xiaorenwu702/p/5672300.html
Copyright © 2020-2023  润新知