Android 自定义阴影

最近在项目碰到一个比较头疼的项目,设计师需要给ui图中的一些按钮之类的东西添加阴影。乍一看设计图,这没啥嘛,咱们Android中不是有这个属性嘛,于是撸起袖子开搞:

<TextView
android:id="@+id/btn_test_performance"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:elevation="5dp"
android:text="@string/hello"
android:background="@drawable/shape_round_white"
android:padding="20dp"
android:layout_marginTop="10dp"
android:layout_gravity="center"/>

图1

代码敲完跑起来看一下效果图,恩,效果还可以。于是把所有需要阴影的控件全部加上android:elevation属性,美滋滋,有api就是好。事情发展得很顺利,直到我们设计师找到我,“你是Android设计师吧,你这个颜色有点不对啊,我们标记的颜色比较淡,你这个颜色有点深”。。。瞬间凌乱,wtf,这个颜色是系统自带的啊,我怎么控制。于是上网google一下,怎样修改elevation的颜色属性。一顿搜索后,无果。祸不单行,我们测试工程师也找了过来,“你这个页面不对啊,我们设计图上这里有阴影,你这里啥也没有”。???什么鬼,不可能啊,我这里都有的,你这个包有问题吧。于是把测试的测试机拿过来自己装上一个一跑。。。尼玛,什么鬼,真没有。回头再看一遍自己的代码发现在android:elevation上有一个黄色的警告,之前没注意,鼠标放上去一看,


图2

。。原来是版本适配问题,看来是api是不能用了,只能通过一些其他的手段去实现了。经过漫长的思考人生后,想出下面几个思路:

  1. 自定义shape.xml drawable文件去给控件设置背景,使用Gradient渐变色属性实现阴影样式。
  2. 找ui设计师,让他们切一个带阴影的图给我们(找ui能解决的都不是事~)
  3. 自己代码实现一个带背景的自定义view

经过不停地自我推翻以及方案被否定后还是留下了第三个方案,主要原因如下:

  • 第一个方案的渐变色不支持四周向中间的渐变,只能支持线性或者环装形式渐变,这两种都不满足需要,因为阴影本身是一个四周一层很淡的颜色包围,在一个矩形框的层面上颜色大概一致,所以这个思路无法实现这个需求,抛弃。
  • 第二个方案询问了一下我们ui。。他们给出的结果是如果使用切图的话那标注的话很难标,身为一个优秀的设计师大多对像素点都和敏感,界面上的像素点有一点不协调那都是无法容忍的。。
  • 那么就只剩下第三个方案了

经过资料查询,发现有两个方式用来写阴影很合适。在我们Android中自定义view的时候有以下两个方法,我们可能用的都不是很多:

  1. paint.setShadowLayer(float radius, float dx, float dy, int shadowColor);
    这个方法可以达到这样一个效果,在使用canvas画图时给视图顺带上一层阴影效果。
    简单介绍一下这几个参数:
    radius: 阴影半径,主要可以控制阴影的模糊效果以及阴影扩散出去的大小。
    dx:阴影在X轴方向上的偏移量
    dy: 阴影在Y轴方向上的偏移量
    shadowColor: 阴影颜色。
    终于找到了设置颜色的,通过设置shadowColor来控制视图的阴影颜色。
    我们试一下使用paint.setShadowLayer(10, 0, 0, Color.RED)的效果:


    图3

    这个是在4.1的设备上的效果图,上面的hello world使用elevation,可以看到丝毫没有效果。下面是使用自定义view的方式画出来的一个图,可以看到效果是很nice的,这个也实现了颜色的可调整性,完美解决。

这里也顺带说下第二种思路,这边我试了一下效果没这边好,有可能是我没调整好,个人觉得应该也是能实现这种效果的, paint还有一个方法是paint.setMaskFilter();这里通过设置BlurMaskFilter()应该也能达到这种效果。

对于第一种方案,经过思考决定封装成一个Viewgroup比较合适,将需要设置阴影的视图放到Viewgroup里面。其中涉及到几个属性,阴影的宽度,view到Viewgroup的距离,如果视图和父布局一样大的话,那阴影就不好显示,如果要能够显示出来就必须设置clipChildren=false。还有就是视图自带的圆角,大部分背景都是有圆角的,比如上图中的圆角,需要达到高度还原阴影的效果就是的阴影的圆角和背景保持一致。
这里把这几个属性抽成自定义属性:

<declare-styleable name="ShadowContainer">
<attr name="containerShadowColor" format="color"/>
<attr name="containerShadowRadius" format="dimension"/>
<attr name="containerDeltaLength" format="dimension"/>
<attr name="containerCornerRadius" format="dimension"/>
<attr name="deltaX" format="dimension"/>
<attr name="deltaY" format="dimension"/>
</declare-styleable>

  • containerShadowColor:设置阴影颜色
  • containerShadowRadius:设置阴影的扩展距离
  • containerDeltaLength:设置子View到ShadowContainer的距离
  • containerCornerRadius:设置子View背景的圆角大小
  • deltaX: 设置阴影向下移动距离
  • deltaY: 设置阴影在Y轴移动距离


    图4

    这里使用画图的方式解释一下这几个参数:
    图中的红色区域是需要设置阴影的View,外部黑色边框包裹的区域是ShadowContainer,中间的白色部分是阴影展示的区域,这块的大小根据设置的containerDeltaLength控制。这里打个比方,如果目标view需要距离左边20dp的话,而中间的containerdeltaLength为10dp,则只需要将ShadowContainer的marginLeft设置为10dp就可以了,这样可以完美控制子view的位置。

此外,containerShadowRadius影响的是阴影在空白区域的占比情况,如果containerShadowRadius小一点的话阴影倒是不大, 但是如果太大的话那么空白区域可能装不下阴影面积。这里需要看情况调整。

使用方式:

<example.chenj.com.apptest.ShadowContainer
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="50dp"
android:layout_gravity="center"
app:containerCornerRadius="5dp"
app:containerDeltaLength="5dp"
app:containerShadowColor="#f00"
app:containerShadowRadius="5dp">
<View
android:layout_width="80dp"
android:layout_height="80dp"
android:background="@drawable/shape_round_white"/>
</example.chenj.com.apptest.ShadowContainer>

效果图:


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

推荐阅读更多精彩内容