美文网首页Android故事Android知识程序员
一周干货总结-(控件状态异常、右优先的一种布局、Fragment

一周干货总结-(控件状态异常、右优先的一种布局、Fragment

作者: 骑着海去看蜗牛 | 来源:发表于2016-12-21 11:12 被阅读400次

    控件状态异常

    当改变顶部Actionbar布局背景颜色的alpha值,以达到Actionbar不透明和全透明效果时,发现其它所有用到这个颜色页面状态都变了。

    //使用如下
    mActionbarView.getBackground().setAlpha(255);
    
    //mActionbarView的布局如下
    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:id="@+id/action_bar"
        android:layout_width="match_parent"
        android:layout_height="@dimen/titlebar_height"
        android:orientation="horizontal"
        android:background="@color/white"
        android:baselineAligned="false">
    </LinearLayout>
    

    可以看到我们给mActionbarView设置了白色的背景色,可问题恰恰就出在这个white的背景色上。
    好吧,开始解释原因。
    在布局中,多个控件共用某一个资源的时候,这些控件会共用一个状态,比如ColorState。如果你改变了一个控件,其它的控件都会接收到相同的通知。所以才会出现上面描述的问题。
    那么问题来了,有办法解决吗?答案是肯定的。
    使用mutate()方法使控件状态不定,这样不定状态的控件就不会共享自己的状态。代码如下

    mActionbarView.getBackground().mutate().setAlpha(255);
    

    右优先的一种布局

    这是一种项目中很常用的需求,需要布局自适应宽度,并且要右侧的View优先显示完,左侧的View如果一行显示不完,则以省略号表示。
    不知道有没有描述清楚,这中布局其实很不好实现,右侧的View需要自适应宽度,并且要优先显示完,然后才显示左边的View的内容,左边的View显示不完了,则省略号,并且左右的View是紧贴着并且自适应宽度的。大家可以想一想该怎么实现这种布局?
    好吧,如果你想不出来,我来告诉你答案吧。

    <RelativeLayout
        android:id="@+id/rl_name"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginLeft="10dp"
        android:layout_marginRight="10dp"
        android:layout_marginTop="16dp"
        android:layout_toLeftOf="@id/tv_follow"
        android:layout_toRightOf="@id/avatar"
        android:gravity="fill">
        <TextView
            android:id="@+id/tv_tag"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_alignParentRight="true"
            android:layout_centerVertical="true"
            android:layout_marginLeft="8dp"
            android:maxLines="1"
            android:textColor="@color/black" 
            android:textSize="@dimen/text_size_24px" />
        <TextView
            android:id="@+id/tv_name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_toLeftOf="@id/tv_tag"
            android:ellipsize="end"
            android:maxLines="1"
            android:textColor="@color/black"
            android:textSize="@dimen/text_size_32px"
            android:textStyle="bold" />
    </RelativeLayout>
    

    答案就是实用android:gravity="fill"这个属性,然后先显示右侧的View,再让左侧的View布局在右侧View的左方即可,你会发现神奇的实现了这种布局效果。

    Fragment动态切换

    关于Fragment的replace方法和add方法
    动态切换显示 Activity 中的多个 Fragment 时,可以通过 replace() 实现,也可以 hide() 和 show() 方法实现。事实上,我们更倾向于使用后者,因为 replace() 方法不会保留 Fragment 的状态,也就是说诸如 EditText 内容输入等用户操作在 remove() 时会消失。当然,如果你不想保留用户操作的话,可以选择前者,视情况而定。

    FragmentManager fm = getSupportFragmentManager();
    FragmentTransaction ft = fm.beginTransaction();
    ft.hide(firstStepFragment);
    if (secondStepFragment==null){
        ft.add(R.id.fl_content, secondStepFragment);
    }else {
        ft.show(secondStepFragment);
    }
    ft.addToBackStack(null);
    ft.commit();
    

    通过 addToBackStack() 保存当前事务
    当用户按下返回键时,如果回退栈中保存有之前的事务,便会执行事务回退,而不是 finish 掉当前 Activity。
    这种方式用在注册登录涉及多个页面的时候会非常合适

    RecyclerView异常处理

    这周在新建一个Activity中使用Recyclerview做一个很简单的字符串展示的时候,出现了一个奇葩的异常。The specified child already has a parent. You must call removeView() on the child's parent first。
    然后指向项目基类重写的一个Layout的layout方法中,百思不得骑姐,找不到哪里出错了,因为是刚接手的项目,之前版本用到了右滑关闭页面的一个库作为基类,所以想到会不会跟这个有关系,但是比较其它的Activity排除了这种猜测,最终发现是自己犯了个小错误。
    在RecyclerView的Adapter中,onCreateViewHolder的时候attach了parent为true导致的。

    when inflating you shouldn't attach the view to its parent. you wrote:
    View v = inflater.inflate(R.layout.my_text_view, parent, true);
    which should be :
    View v = inflater.inflate(R.layout.my_text_view, parent, false);
    

    代码修改为如下完美运行,瞬间石化

    @Override
    public UserDetailViewHolder onCreateDataItemViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_user_detail,
                parent, false);
        UserDetailViewHolder userDetailViewHolder = new UserDetailViewHolder(view);
        return userDetailViewHolder;
    }
    

    相关文章

      网友评论

      • ziabo_yu:第一个需求,可以用LinearLayout来做的,oritation:horizontal 右边的控件wrap,左边width=0dp,weight=1,也是完美解决的,中间的回退栈不错,最后那个recycleView很少用,不做评价
        骑着海去看蜗牛: @ziabo_yu linearlayout match_parent肯定不行,wrap_content没有验证

      本文标题:一周干货总结-(控件状态异常、右优先的一种布局、Fragment

      本文链接:https://www.haomeiwen.com/subject/bnifvttx.html