Android 开发中不得不知道的 Tips 集合

1.你还在写Drawable来实现Imageview的点击效果?

很多时候我们需要给ImageView添加点击效果,例如title上的back按钮。


通常来讲,UI那边会给我们两张图,一张选中效果,一张nomal效果;我们会风骚的撸一个Drawable

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <!-- 按压时 -->
    <item android:drawable="@mipmap/btn_enter_pressed" android:state_pressed="true" />
    <!-- 默认时 -->
    <item android:drawable="@mipmap/btn_enter_normal" />
</selector>

然后在布局文件中把Imageview的background属性设置成你写的Drawable文件。例如:

<ImageView
        android:id="@+id/tv_login"
        android:layout_width="60dp"
        android:layout_height="60dp"
        android:background="@drawable/back_click" />

这样当然没问题,毕竟都是大家熟悉的套路。不料,你突然接到了一个需求,为了支持动态换肤,这个back的图片需要从网络上获取,并且仍然需要支持点击效果。顿时,无数程序猿心中众多那个啥在奔腾。

解决方案:
继承ImageView,监听OnTouchListener的事件,动态设置setColorFilter

public class ClickImageView extends AppCompatImageView {

    public ClickImageView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    public ClickImageView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init();
    }

    public ClickImageView(Context context) {
        super(context);
        init();
    }

    private void init() {
        setOnTouchListener(onTouchListener);
    }

    private OnTouchListener onTouchListener = new OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            switch (event.getAction()) {
                case MotionEvent.ACTION_UP:
                    setColorFilter(null);
                    break;
                case MotionEvent.ACTION_DOWN:
                    changeLight();
                    break;
                case MotionEvent.ACTION_MOVE:
                    break;
                case MotionEvent.ACTION_CANCEL:
                    setColorFilter(null);
                    break;
                default:
                    break;
            }
            return false;
        }
    };

    private void changeLight() {
        int brightness = -80;
        ColorMatrix matrix = new ColorMatrix();
        matrix.set(new float[]{1, 0, 0, 0, brightness, 0, 1, 0, 0,
                brightness, 0, 0, 1, 0, brightness, 0, 0, 0, 1, 0});
        setColorFilter(new ColorMatrixColorFilter(matrix));
    }
}

布局文件中这么搞妥了

    <你的包名.ClickImageView
        android:id="@+id/iv_share"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:src="@mipmap/act_ic_share" />

2.WebView加载视频or音频时候的二次元世界

现在APP里面怎么能少的了WebView的舞台呢?不过加载如下网页的时候会有坑


没错,这个网页里面有视频,用户播放视频,然后点击了返回键,此时如果直接finish掉当前的WebActivity时会出现灵异的现象:刚才看的视频仍然在播放,仍然会有声音发出。除非你exit掉咱们的App。

解决方案:
在WebActivity中控制一下WebView,亲测有效

    @Override
    protected void onResume() {
        super.onResume();
        wb_content.onResume();
    }

     @Override
    protected void onDestroy() {
        super.onDestroy();
        wb_content.destroy();  //手动销毁WebView
    }

    @Override
    protected void onPause() {
        super.onPause();
        wb_content.onPause();
    }

3.是时候从Rxjava1换到Rxjava2啦

还没玩过Rxjava的同学们建议直接从Rxjava2学起,现在还在奋斗在Rxjava1的同学们建议尽快转到Rxjava2的战线。Rxjava1很快就停止更新了。废话不多说,直接祭出官方wiki
https://github.com/ReactiveX/RxJava/wiki/What's-different-in-2.0

4.控制Recyclerview滑动的问题

很多场景下,产品需要我们通过代码控制Recyclerview滑动到第几个position,例如:用户下拉刷新当天节目列表,我们应该计算当前时间播放的是第几个节目,然后滑动到这个position,注:此时这个position应该居于屏幕的中间

解决方案:
这里只说一下LinearLayoutManager下的解决方式

public class CenterLayoutManager extends LinearLayoutManager {

    public CenterLayoutManager(Context context) {
        super(context);
    }

    public CenterLayoutManager(Context context, int orientation, boolean reverseLayout) {
        super(context, orientation, reverseLayout);
    }

    public CenterLayoutManager(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
    }

    @Override
    public void smoothScrollToPosition(RecyclerView recyclerView, RecyclerView.State state, int position) {
        RecyclerView.SmoothScroller smoothScroller = new CenterSmoothScroller(recyclerView.getContext());
        smoothScroller.setTargetPosition(position);
        startSmoothScroll(smoothScroller);
    }

    private static class CenterSmoothScroller extends LinearSmoothScroller {

        CenterSmoothScroller(Context context) {
            super(context);
        }

        @Override
        public int calculateDtToFit(int viewStart, int viewEnd, int boxStart, int boxEnd, int snapPreference) {
            return (boxStart + (boxEnd - boxStart) / 2) - (viewStart + (viewEnd - viewStart) / 2);
        }
    }
}

然后使用Recyclerview的时候,设置LayoutManager为CenterLayoutManager。需要滚动到第几个item直接调用

recyclerview.smoothScrollToPosition(position);

就妥啦。效果如下。




1.sp还是dp?

众所周知,官方建议我们字体的单位使用sp,这样用户在“系统设置”中调整了系统字体大小的时候,我们app中的字体会随着系统字体的大小而改变。So,众猿机智的在布局文件中写下了以下代码

    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="this is text"
        android:textSize="15sp" />

当然上述TextView不会有什么问题,因为这货height是自适应的。但是很多情境下,例如ListView或者Recyclerview的item中,高度是固定的时候,sp就会有适配问题,例如


解决方案:
在Application中重写onConfigurationChanged 强制字体不随着系统改变而改变(微信也是这么干的)

@Override
    public void onConfigurationChanged(Configuration newConfig) {
        if (newConfig.fontScale != 1)//非默认值
            getResources();
        super.onConfigurationChanged(newConfig);
    }

    @Override
    public Resources getResources() {
        Resources res = super.getResources();
        if (res.getConfiguration().fontScale != 1) {//非默认值
            Configuration newConfig = new Configuration();
            newConfig.setToDefaults();
            //设置默认
            res.updateConfiguration(newConfig, res.getDisplayMetrics());
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
                createConfigurationContext(newConfig);
            } else {
                res.updateConfiguration(newConfig, res.getDisplayMetrics());
            }
        }
        return res;
    }

2.使用PhotoView+Viewpager崩溃问题

这几天碰到一个诡异的问题,PhotoView+Viewpager开发图集效果的时候,在三星Galxy系列手机上手指放大的时候没问题,手指捏合的时候出现java.lang.IllegalArgumentException: pointerIndex out of range 异常然后闪退。Google了一波,说是三星系统的Bug。。。我等应用层开发汪总不见得给每个三星用户修改一下底层代码吧。

解决方案:
自定义一个Viewpager,重写onInterceptTouchEvent函数,在里面捕获IllegalArgumentException就妥了。

public class PhotoViewPager extends android.support.v4.view.ViewPager {
    public PhotoViewPager(Context context) {
        super(context);
    }

    public PhotoViewPager(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        try {
            return super.onTouchEvent(ev);
        } catch (IllegalArgumentException ex) {
            ex.printStackTrace();
        }
        return false;
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        try {
            return super.onInterceptTouchEvent(ev);
        } catch (IllegalArgumentException ex) {
            ex.printStackTrace();
        }
        return false;
    }
}

布局文件中用PhotoViewPager代替系统原生的ViewPager就好啦。

3.什么?DatePickerDialog有Bug?

项目中用到了修改用户生日的功能,打算直接用DatePickerDialog来做一个时间选择起,然后诡异的事情发生了。用户选择完时间之后onDateSelect的回调函数竟然执行了两次。原来4.1跟4.2版本有个系统Bug,这时候我们需要重新定义一个DatePickerDialog来屏蔽onStop方法。


public class RepairDatePickDialog extends DatePickerDialog {
    public RepairDatePickDialog(Context context, OnDateSetListener callBack,
                                int year, int monthOfYear, int dayOfMonth) {
        super(context, callBack, year, monthOfYear, dayOfMonth);
        // TODO Auto-generated constructor stub
    }

    public RepairDatePickDialog(Context context, int theme,
                                OnDateSetListener callBack, int year, int monthOfYear,
                                int dayOfMonth) {
        super(context, theme, callBack, year, monthOfYear, dayOfMonth);
        // TODO Auto-generated constructor stub
    }

    protected void onStop() {
        // TODO Auto-generated method stub
    }
}

4.浏览器中打开自家App的那些套路

大体的需求是用户分享出去连接,被分享的用户在浏览器中打开此连接,如果该用户设备上有我们的app则吊起app,如果没有则通知用户去下载我们的app
之前写过一片博客,详细的介绍了这种Deeplink的实现方式,请移步
https://juejin.im/entry/590fe2d8ac502e006cf9e3e4/detail
github地址
https://github.com/weixinjie/DeepLink

5.Android路由机制浅析

大厂的客户端里面都用了路由来实现页面之前的跳转,引入路由机制并不全是为了页面之间的解耦合,更多的是为了配合运营的套路。例如:大部分app里面都有Banner,这个Banner不一定全是打开Webview。有的Banner item打开的是充值页面,有的Banner item打开的是用户详情页面等。当然你可以为了Banner来写一个Map集合,每次用户点击Banner的时候Switch一下type,然后打开相应的页面,但是如果用户在浏览器中Deeplink到本地客户端呢?是不是也要实现一套Map?如果是在Webview中与js交互打开各个页面呢?是不是还要实现一套Map?

推荐一个很好用的路由框架
https://github.com/mzule/ActivityRouter
后期我会献上这个库的源码解析。


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值