美文网首页Android适配
Android5.0~7.0特性以及适配

Android5.0~7.0特性以及适配

作者: Android大混子 | 来源:发表于2017-04-19 16:14 被阅读868次

    android-5.0

    1.Design Library

    CardView

    RecyclerView

    ToolBar

    FloatingActionButton

    TextInputLayout

    TabLayout

    NavigationView

    CoordinatorLayout

    AppBarLayout

    CollapsingToolbarLayout

    NestedScrolling

    Snackbar

    2.动态替换Theme

    MaterialTheme配色方案:www.materialpalette.com 

    修改状态栏,ActionBar,界面背景,NavigationBar的颜色。让Activity使用自定义的Theme。

    @color/colorWindowBackground

    动态替换Theme的步骤:

    定义至少2套theme

    调用setTheme方法设置当前的theme,但是该方法要在setContentView之前,如:

    setTheme(mTheme);

    setContentView(R.layout.activity_main);

    设置了Theme,需要finish当前Activity,然后重启当前Activity,让Theme生效

    Intent intent = getActivity().getIntent();

    getActivity().finish();//结束当前的Activity

    getActivity().overridePendingTransition(0,0);//不要动画

    startActivity(intent);

    3.View的高度与阴影

    官网介绍:https://developer.android.com/intl/zh-tw/training/material/shadows-clipping.html 

    View新增属性z轴,用来体现Material Design中的层次,影响因素2个:elevation和translationZ

    View高度= elevation + translationZ

    elevation表示view的高度,高度越大,阴影越大,可以在xml中直接使用属性, 也可以在代码中使用view.setEvelvation();

    android:elevation="10dp"

    transtionZ属性表示view在Z方向移动的距离,一般用于属性动画中

    android:translationZ="10dp"

    高度影响View的绘制顺序,过去是按View添加顺序绘制,先添加的先绘制,现在高度小的先绘制,因为高度小的,层级低,在下面, 高度相同的,按添加顺序绘制

    注意:

    如果View的背景色为透明,则不会显示出阴影效果

    只有子View的大小比父View小时,阴影才能显示出来

    4.View的轮廓与裁剪(在Android5.1以及以上才有效果)

    官网介绍:https://developer.android.com/intl/zh-tw/training/material/shadows-clipping.html 

    View增加了轮廓概念,轮廓用来表示怎么显示阴影,也就是说轮廓什么形状,阴影就显示什么形状。

    View的轮廓可以通过outlineProvider属性设置,默认是依据于background的,还有其他3个取值:bounds,none,paddingBounds

    android:outlineProvider="bounds"

    none:即使设置了evaluation也不显示阴影

    background:按背景来显示轮廓,如果background是颜色值,则轮廓就是view的大小,如果是shape,则按shape指定的形状作为轮廓

    bounds: View的矩形大小作轮廓

    paddedBounds: View的矩形大小减去padding的值后的大小作轮廓。

    可以通过setOutlineProvider()方法自定义轮廓:

    tv_blue.setOutlineProvider(new ViewOutlineProvider() {

    @TargetApi(Build.VERSION_CODES.LOLLIPOP)

    @Override

    public void getOutline(View view, Outline outline) {

    outline.setOval(0,0,

    view.getWidth(),view.getHeight());

    }

    });

    注意:如果background是图片,那只能通过代码setOutlineProvider()来指定轮廓

    View的裁剪是指将View按照轮廓裁剪,能改变View的形状,如圆形头像:

    先设置轮廓:

    再设置根据轮廓裁剪View,目前只支持对矩形,圆形,圆角矩形的裁剪:

    //设置对View进行裁剪

    tv_clip.setClipToOutline(true);

    5.Palette的使用

    使用Palette可以让我们从一张图片中拾取颜色,将拾取到的颜色赋予ActionBar,StatusBar以及背景色可以让界面色调实现统一

    使用Palette需要添加以下依赖:compile 'com.android.support:palette-v7:23.0.0+'

    Palette提供的API传入Bitmap即可获取Palette对象,以下是同步和异步使用方式:

    //同步获取,需要在子线程中使用

    Palette palette = Palette.from(drawable.getBitmap()).generate();

    //异步获取,可以在主线程中使用

    Palette.from(drawable.getBitmap()).generate(new Palette.PaletteAsyncListener() {

    @Override

    public void onGenerated(Palette palette) {

    }

    });

    得到Palette对象后,获取其中的颜色,颜色对应如下:

    vibrant      -有活力的颜色

    lightVibrant -有活力的亮色

    darkVibrant  -有活力的暗色

    muted        -柔和暗淡的颜色

    lightMuted  -柔和的亮色

    darkMuted    -柔和的暗色

    获取指定颜色的采样对象,获取采样得到的颜色:

    //我们可以直接使用palette获取指定颜色:

    palette.getLightMutedColor(defaultColor);

    //一般也可以先获取采样对象Swatch,从Swatch中获取我们需要的颜色:

    //获取有活力颜色的采样对象

    Palette.Swatch vibrantSwatch = palette.getVibrantSwatch();

    采样对象Swatch提供了以下方法来获取颜色:

    //swatch.getPopulation(): the amount of pixels which this swatch represents.

    //swatch.getRgb(): the RGB value of this color.

    //swatch.getHsl(): the HSL value of this color,即色相,饱和度,明度.

    //swatch.getBodyTextColor(): the RGB value of a text color which can be displayed on top of this color.

    //swatch.getTitleTextColor(): the RGB value of a text color which can be displayed on top of this color

    //一般会将getRgb设置给控件背景色,getBodyTextColor()设置给文字颜色

    textView.setBackgroundColor(vibrantSwatch.getRgb());

    textView.setTextColor(vibrantSwatch.getBodyTextColor());

    6.水波纹动画,自定义水波纹动画以及状态选择器动画

    首先,在Android5.0以上,点击效果默认自带水波纹效果,并且有2种选择:

    //矩形边框水波纹

    android:background="?android:attr/selectableItemBackground"

    //无边框限制水波纹

    android:background="?android:attr/selectableItemBackgroundBorderless"

    自定义水波纹动画

    使用ViewAnimationUtils创建圆形水波纹动画,注意该动画不能在Activity的onCreate方法中执行:

    Animator circularReveal = ViewAnimationUtils.createCircularReveal(text, 0, text.getHeight() , 1f, text.getWidth()*2);

    circularReveal.setDuration(1000);

    circularReveal.start();

    使用ripple标签或者RippleDrawable可以更改控件水波纹动画颜色:

    定义带有属性动画的状态选择器

    通过stateListAnimator属性指定状态选择器的动画:

    android:stateListAnimator="@drawable/selector_anim"

    状态选择器文件中需要加入objectAnimator标签:

    android:duration = "@android:integer/configshortAnimTime"

    android:valueTo = "0.2"

    android:valueFrom = "1"

    android:valueType = "floatType" >

    同样,状态选择器动画可以用代码方式加载

    //加载动画

    AnimatorInflater.loadStateListAnimator();

    //设置动画

    View.setStateListAnimator();

    定义带有帧动画的状态选择器,需要设置给background属性,不是stateListAnimator,如下所示:

    android:state_pressed = "true" />

    android:drawable = "@drawable/drawableD" />

    7. service的注册必须显示注册,不能隐式注册,相关链接

    http://www.eoeandroid.com/thread-568853-1-1.html 

    现象:Service Intent must be explicit:

    解决:intent.setPackage("XXXXX");

    8.android5.0art运行报错:

    http://www.eoeandroid.com/forum.php?mod=viewthread&tid=564427 

    9.webview坑

    It's beceause of Cookie Policy, to fix it, you should add this :

    public void onCreate(Bundle savedInstanceState)

    {

    super.onCreate(savedInstanceState);

    super.init();

    // Allow third party cookies for Android Lollipop

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {

    WebView webView = (WebView)super.appView;

    CookieManager cookieManager = CookieManager.getInstance();

    cookieManager.setAcceptThirdPartyCookies(webView,true);

    }

    super.loadUrl(Config.getStartUrl());

    }

    if you are using Android Lollipop, then

    CookieManager.getInstance().setAcceptCookie(true);

    won't work. You need to use

    CookieManager.getInstance().setAcceptThirdPartyCookies(true);

    10.notification适配

    http://blog.csdn.net/sk719887916/article/details/40541143 

    11.JobScheduler来执行一些需要满足特定条件但不紧急的后台任务,APP利用JobScheduler来执行这些特殊的后台任务时来减少电量的消耗。

    12.AnimationDrawable在5.0及以上已优化但尽量不要使用,它在初始化时就将所有图片加载到内存中,特别占内存,并且不能释放,释放后下次加载时会报错

    13.三星5.0-5.1机型上DatePicker控件会崩溃。MIUI上SYSTEM_ALERT_WINDOW需要额外申请权限,默认会拒绝掉,除非type设置为TYPE_TOAST。

    14.5.0以上关闭通知权限弹不出toast http://blog.csdn.net/qq_25867141/article/details/52807705 

    15.FlashLight按照以前的接口是不能用了,注意是不能用了,所以一定要做版本适配,Lollipop上把Camera直接杠掉,从新弄了一套Camera2,

    详情可以去看Lollipop状态栏闪光灯开关的源码https://android.googlesource.com/platform/frameworks/base/+/android-5.0.0_r2/packages/SystemUI/src/com/android/systemui/statusbar/policy/FlashlightController.java 

    16.ActivityManger中通过getRunningTask获取当前列表也是禁用了,这个是代码上注释的描述

    @deprecated As of {@link android.os.Build.VERSION_CODES#LOLLIPOP},

    this method is no longer available to third party applications:

    the introduction of document-centric recents means it can leak person information to the caller.

    For backwards compatibility, it will still return a small subset of its data: at least the caller's own tasks, and possibly some other tasks such as home that are known to not be sensitive.

    这就禁止后我就无法获取到当前运行的packageName了,也就没法进行一些涉及到需求的判断了。目前我还没找到很好的解决方案,涉及到系统设计的问题,可能只能从产品上去改进了

    17.支持Vector图片(SVG标准)

    18.元素共享及兼容4.x

    http://blog.csdn.net/u012342082/article/details/50599596 

    19.状态栏沉浸式

    http://niorgai.github.io/2016/03/20/Android-transulcent-status-bar/ 

    android-6.0

    1.同一个APP在api <=22的sdk情况下编译,可以运行正常,不存在闪退或者.so库加载失败的情况,当你采用api >=23的sdk编译的时候,

    安装到Android 6.0及其以上的手机的时候,大范围出现崩溃 或者.so库加载失败,而在6.0以下的手机却正常。Catch的信息:dlopen failed:

    cannot locate symbol "XXXX" xxxx.so。

    解决方案:主要有两种:

    1-委曲求全,指标不治本,把你的APK编译时API降低到23以下,还出问题就继续降低,这意味着,你很多Android Sdk的新控件用不了;

    2-在Application.mk中修改APP_STL,重新编译.so,如果,我说如果你没有源码,那么悲剧了,要么等他们解决,要么采用第一种,建议尝试,

    APP_STL := gnustl_shared,这种方式,对于所需要的外部动态链接函数、符号,在NDK 13b中都会独立生成一份,全部引用就解决此类问题,例如

    private void load() {

    try {

    System.loadLibrary("gnustl_shared");

    System.loadLibrary("speex");

    }

    catch (Throwable e) { }

    }

    2.运行时权限处理

    https://www.aswifter.com/2015/11/04/android-6-permission/

    3.使用SYSTEM_ALERT_WINDOW绘制的悬浮窗不能含有elevation属性,否则会在真机上崩溃,模拟器正常。

    android-7.0

    1.DiffUtil RecyclerView帮助类

    2.多窗口模式

    3.ConstraintLayout增强RelativeLayout解决多层嵌套问题

    4.适配

    http://www.devio.org/2016/09/28/Android7.0%e9%80%82%e9%85%8d%e5%bf%83%e5%be%97/ 

    https://mp.weixin.qq.com/s?__biz=MzA5MzI3NjE2MA==&mid=2650239036&idx=1&sn=cdde2a1f6125c8f3d780eff8c8e36076&chksm=88638153bf1408455d5c56f0778d40ba9d4448a1e68ef0bf889c4a99699353e3e4ad99f95398&scene=0#rd 

    1-目录被限制访问 间接导致Notification的点击无效的问题

    私有文件的文件权限不在放权给所有的应用,使用MODE_WORLD_READABLE或MODE_WORLD_WRITEABLE进行的操作将触发SecurityException

    给其他应用传递file:// URI类型的Uri,可能会导致接受者无法访问该路径。 因此,在Android7.0中尝试传递file:// URI会触发FileUriExposedException。

    DownloadManager不再按文件名分享私人存储的文件。COLUMN_LOCAL_FILENAME在Android7.0中被标记为deprecated, 旧版应用在访问COLUMN_LOCAL_FILENAME时可能出现无法访问的路径。

    面向Android N或更高版本的应用在尝试访问COLUMN_LOCAL_FILENAME时会触发SecurityException。

    2-应用间共享文件

    在Android7.0系统上,Android框架强制执行了StrictMode API政策禁止向你的应用外公开file:// URI。 如果一项包含文件file:// URI类型 的Intent离开你的应用,应用失败,

    并出现FileUriExposedException异常,如调用系统相机拍照,或裁切照片

    3-使用FileProvider

    5.PackageManager增加了3个abstract方法,如果继承这个类,需要重写下面的方法,否则在三星的有些机器上会crash

    getPackageGids(String packageName, int flags)

    getPackageUid(String packageName, int flags)

    hasSystemFeature(String name, int version)

    6.Activity的performStop增加了boolean参数,需要反射调用的时候要注意区分

    7.在Android 6.0及以下设置textScaleX为0是没有问题的,在7.1的时候,这个属性设置为>0能正常显示,但是到设置为0的时候就挂了。

    android-8.0

    1.移除HttpsURLConnection中不安全的TLS版本回退

    http://developers.googleblog.cn/2017/04/android-o-httpsurlconnection-tls.html 

    相关文章

      网友评论

        本文标题:Android5.0~7.0特性以及适配

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