Android 自定义Drawable
1.使用BitmapShader实现图片圆角
public class CornerDrawable extends Drawable {
private Paint mPaint;
private Bitmap bmp;
private RectF rectF; public CornerDrawable(Bitmap bmp) {
this.bmp = bmp;
BitmapShader shader = new BitmapShader(bmp, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);
// CLAMP 拉伸
// REPEAT 重复
// MIRROR 镜像
// BitmapShader是从画布的左上角开始绘制的
mPaint = new Paint();
mPaint.setAntiAlias(true);
mPaint.setShader(shader);
} @Override
public void draw(Canvas canvas) {
Rect rect = getBounds();
// Log.e(getClass().getSimpleName(), rect.left + ":" + rect.width() + ":" + rect.height());
// Log.e(getClass().getSimpleName(), rectF.left + ":" + rectF.width() + ":" + rectF.height());
canvas.drawRoundRect(rectF, 20, 20, mPaint);
} @Override
public void setAlpha(int alpha) {
mPaint.setAlpha(alpha);
} @Override
public void setColorFilter(ColorFilter cf) {
mPaint.setColorFilter(cf);
} @Override
public int getOpacity() {
return PixelFormat.TRANSLUCENT;
} // getIntrinsicWidth、getIntrinsicHeight主要是为了在View使用wrap_content的时候,提供一下尺寸
@Override
public int getIntrinsicHeight() {
return bmp.getHeight();
} @Override
public int getIntrinsicWidth() {
return bmp.getWidth();
} @Override
public void setBounds(int left, int top, int right, int bottom) {
super.setBounds(left, top, right, bottom);
rectF = new RectF(left, top, right, bottom);
}
}
2.除了圆角外,还可以指定画图片的某圆弧对应的内容
重写上面的draw方法如下
RectF rf = new RectF(-100, -130, 160, 130);
canvas.drawArc(rf, 0, 120, true, paint);
3.使用PorterDuffXfermode
@Override
public void draw(Canvas canvas) {
PorterDuffXfermode pdf = new PorterDuffXfermode(PorterDuff.Mode.SRC_IN);
paint.drawBitmap(dst, 0, 0, paint);
paint.setXfermode(pdf);
paint.setColor(0xffff4400);
canvas.drawBitmap(src, 0, 0, paint);
paint.setXfermode(null);
}
http://www.jianshu.com/p/d11892bbe055
》》XferMode
- AvoidXfermode 指定了一个颜色和容差,强制Paint避免在它上面绘图(或者只在它上面绘图)。
- PixelXorXfermode 当覆盖已有的颜色时,应用一个简单的像素异或操作。
- PorterDuffXfermode 这是一个非常强大的转换模式,使用它,可以使用图像合成的16条Porter-Duff规则的任意一条来控制Paint如何与已有的Canvas图像进行交互。
》》PorterDuff.Mode为枚举类,一共有16个枚举值:
- PorterDuff.Mode.CLEAR 所绘制不会提交到画布上。
- PorterDuff.Mode.SRC 显示上层绘制图片
- PorterDuff.Mode.DST 显示下层绘制图片
- PorterDuff.Mode.SRC_OVER 正常绘制显示,上下层绘制叠盖。
- PorterDuff.Mode.DST_OVER 上下层都显示。下层居上显示。
- PorterDuff.Mode.SRC_IN 取两层绘制交集。显示上层。
- PorterDuff.Mode.DST_IN 取两层绘制交集。显示下层。
- PorterDuff.Mode.SRC_OUT 取上层绘制非交集部分。
- PorterDuff.Mode.DST_OUT 取下层绘制非交集部分。
- PorterDuff.Mode.SRC_ATOP 取下层非交集部分与上层交集部分
- PorterDuff.Mode.DST_ATOP 取上层非交集部分与下层交集部分
- PorterDuff.Mode.XOR 异或:去除两图层交集部分
- PorterDuff.Mode.DARKEN 取两图层全部区域,交集部分颜色加深
- PorterDuff.Mode.LIGHTEN 取两图层全部,点亮交集部分颜色
- PorterDuff.Mode.MULTIPLY 取两图层交集部分叠加后颜色
- PorterDuff.Mode.SCREEN 取两图层全部区域,交集部分变为透明色
参考:ApiDemos/Graphics/XferModes
4.对图片进行颜色转换
public static Drawable getPrimaryDrawable(int resId) {
Drawable icon = context.getResources().getDrawable(resId);
int baseColor = context.getResources().getColor(R.color._secondary_color);
icon.setColorFilter(baseColor, PorterDuff.Mode.SRC_IN);
return icon;
}
5.通过xml定义drawable
<?xml version="1.0" encoding="utf-8"?>
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
<item>
<shape>
<solid android:color="#ff80cbc4" />
</shape>
</item>
<item android:top="48dp">
<bitmap
android:gravity="center"
android:src="@drawable/app_background_png"
android:tileMode="disabled" />
</item>
</layer-list>
6.自定义按钮状态
<declare-styleable name="CustomStateDrawableButton">
<attr name="state_readed" format="boolean" />
</declare-styleable>
定义一个状态
public class CustomStateDrawableButton extends ImageButton {
public static final int[] MessageReaded = {R.attr.state_readed};
private boolean isReaded = false; public CustomStateDrawableButton(Context context) {
super(context);
} public CustomStateDrawableButton(Context context, AttributeSet attrs) {
super(context, attrs);
} public CustomStateDrawableButton(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
} public void setReaded(boolean isReaded) {
if (this.isReaded != isReaded) {
this.isReaded = isReaded;
//
refreshDrawableState();
}
} @Override
public int[] onCreateDrawableState(int extraSpace) {
if (!isReaded) {
int[] ds = super.onCreateDrawableState(extraSpace + 1);
mergeDrawableStates(ds, MessageReaded);
return ds;
}
return super.onCreateDrawableState(extraSpace);
}
}
// http://www.devdiv.com/Android-Android%E4%B8%ADDrawable%E5%88%86%E7%B1%BB%E6%B1%87%E6%80%BB%EF%BC%88%E4%B8%8A%EF%BC%89-thread-126853-1-1.html
// ColorDrawable、GradientDrawable、BitmapDrawable、 NinePatchDrawable、InsetDrawable、ClipDrawable、ScaleDrawable、RotateDrawable、AnimationDrawable、LayerDrawable、LevelListDrawable、StateListDrawable、TransitionDrawable
Android 自定义Drawable的更多相关文章
- Android自定义drawable(Shape)详解
在Android开发过程中,经常需要改变控件的默认样式, 那么通常会使用多个图片来解决.不过这种方式可能需要多个图片,比如一个按钮,需要点击时的式样图片,默认的式样图片. 这样就容易使apk变大. 那 ...
- Android APK开发 Drawable文件夹下的自定义Drawable文件
版本:2018/2/11 Drawable的分类 自定义Drawable SVG矢量图 个人总结的知识点外,部分知识点选自<Android开发艺术探索>-第六章 Drawable 1.Dr ...
- Android 使用自定义Drawable 设置圆角矩形或者圆形图片
转自 Android Drawable 那些不为人知的高效用法 本文出自:[张鸿洋的博客] http://blog.csdn.net/lmj623565791/article/details/437 ...
- Android—自定义开关按钮实现
我们在应用中经常看到一些选择开关状态的配置文件,做项目的时候用的是android的Switch控件,但是感觉好丑的样子………… 个人认为还是自定义的比较好,先上个效果图:
- android 自定义通知栏
package com.example.mvp; import cn.ljuns.temperature.view.TemperatureView;import presenter.ILoginPre ...
- 最简单的android自定义进度条样式
一.自定义圆形进度条样式 1.在安卓项目drawable目录下新建一个xml文件如下:<?xml version="1.0" encoding="utf-8&quo ...
- Android自定义spinner下拉框实现的实现
一:前言 本人参考博客:http://blog.csdn.net/jdsjlzx/article/details/41316417 最近在弄一个下拉框,发现Android自带的很难实现我的功能,于是去 ...
- Android自定义对话框
在android中有自带的对话框,为了美观,很多开发者会使用自定义对话框,如下图: 点击“弹出自定义对话框按钮后”显示如图效果. 首先要自己定义一个xml文件定义自己对话框的样式: <?xml ...
- android 自定义Style初探---ProgressBar
系统自带的ProgressBar太丑了,所以我决定自定义一个Style. 原来的Style <?xml version="1.0" encoding="utf-8& ...
随机推荐
- [游戏模版10] Win32 平面地图贴图 正
>_<:picture resource >_<:If you master the ways of mapping picture,then this problem is ...
- 通俗易懂的深入理解js闭包
闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域 ...
- C#课外实践——校园二手平台(技术篇3)
说明:生活中,又有谁,能真正摆脱周围环境的束缚,而追随自己的内心呢? ListView的简单用法. 最后展示几张效果图吧 主窗体 登录窗体,虽然没有角色 选择,但已经隐藏在代码里了. 选择购买窗体,这 ...
- Qt之Dialog\widget\ mainwindow的区别和布局管理器 & 分裂器的区别
1.Dialog\widget\ mainwindow的区别 注意mainwindow和widget的区别,mainwindow都工具栏和菜单栏 Dialog and mainwinodws 都是继承 ...
- crossplatform---bower解决js的依赖管理
从零开始nodejs系列文章,将介绍如何利Javascript做为服务端脚本,通过Nodejs框架web开发.Nodejs框架是基于V8的引擎,是目前速度最快的Javascript引擎.chrome浏 ...
- js不是从上到下执行的吗?
如果说js是从上到下解释执行的, 那么,按道理应该会执行错误前面的代码. 如: [代码一] //输出1,2,到3报错 console.log("一") console.log(&q ...
- JavaScript 语句 for
循环for语句: for(var i =1;i<=5;i++)(初始条件:循环条件:状态改变) { 循环体 } 循环的作用:反复执行某段代码 四要素:初始条件.循环条件.循环体.状态改变 例题1 ...
- VC2010 调用 Webservice
开发环境:VC2010,gsoap_2.8.23 http://blog.csdn.net/zhaiwenjuan/article/details/6590941 使用soapcpp2的时候要加参数- ...
- 友好解决POI导入Excel文件行是不是为空
继 解决POI读取Excel如何判断行是不是为空 后发现了一个问题.这个是一个银行的需求,有20万个客户的资料要导入系统,但有的资料是有问题的(不能正常导入),但也有能正常导入的.现在的问题是怎么知道 ...
- latextools \cite 自动补全
最近在用latex写毕业论文,编辑环境用的是Sublime Text 2 加 latextools 插件,在使用latextools的\cite命令来引用参考文献时,我们希望输入\cite{ 后自动弹 ...