开发中遇到美工出的图标是外发光,确实好看,可直接切外发光的图标,效果没有设计图上好,那么只能Android自己实现了.
自定义View
/**
* 设置图标外发光
* Created by boyko on 2017/5/11.
*/
public class LightView extends View {
private Paint mPaint;
private Rect mSrcRect;
private Rect mDestRect;
private Bitmap bmp;
private Bitmap shadowBitmap;
private int width;
private int height;
public LightView(Context context) {
super(context);
init(context, null);
}
public LightView(Context context, AttributeSet attrs) {
super(context, attrs);
init(context, attrs);
}
public LightView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init(context, attrs);
}
private void init(Context context, AttributeSet attrs) {
// 关闭硬件加速
setLayerType(LAYER_TYPE_SOFTWARE, null);
mPaint = new Paint();
TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.LightView, 0, 0);
int n = a.getIndexCount();
Drawable src_resource;
for (int i = 0; i < n; i++) {
int attr = a.getIndex(i);
switch (attr) {
case R.styleable.LightView_light_color:
mPaint.setColor(a.getColor(attr, Color.WHITE));
break;
case R.styleable.LightView_light_radius:
mPaint.setMaskFilter(new BlurMaskFilter(a.getFloat(attr, 5f), BlurMaskFilter.Blur.OUTER));
mPaint.setAlpha(180); // 可根据实际情况调整
break;
case R.styleable.LightView_light_src:
src_resource = a.getDrawable(attr);
bmp = drawableToBitmap(src_resource);
break;
}
}
}
@SuppressLint("DrawAllocation")
@TargetApi(Build.VERSION_CODES.KITKAT)
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
width = this.getMeasuredWidth();
height = this.getMeasuredHeight();
// 获取位图的Alpha通道图
shadowBitmap = Bitmap.createScaledBitmap(bmp.extractAlpha(), width, height, true);
bmp = Bitmap.createScaledBitmap(bmp, width, height, true);
mSrcRect = new Rect(0, 0, shadowBitmap.getWidth(), shadowBitmap.getHeight());
mDestRect = new Rect(0, 0, shadowBitmap.getWidth(), shadowBitmap.getHeight());
canvas.drawBitmap(shadowBitmap, mSrcRect, mDestRect, mPaint);
canvas.drawBitmap(bmp, mSrcRect, mSrcRect, null);
}
public Bitmap drawableToBitmap(Drawable drawable) {
Bitmap bitmap = Bitmap.createBitmap(
drawable.getIntrinsicWidth(),
drawable.getIntrinsicHeight(),
drawable.getOpacity() != PixelFormat.OPAQUE ? Bitmap.Config.ARGB_8888: Bitmap.Config.RGB_565);
Canvas canvas = new Canvas(bitmap);
drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());
drawable.draw(canvas);
return bitmap;
}
/**
* 重写onMeasure,解决在wrap_content下与match_parent效果一样的问题
* @param widthMeasureSpec
* @param heightMeasureSpec
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
int height = width;
int width = height;
final int widthSpecMode = MeasureSpec.getMode(widthMeasureSpec);
final int widthSpecSize = MeasureSpec.getSize(widthMeasureSpec);
final int heightSpecMode = MeasureSpec.getMode(heightMeasureSpec);
final int heightSpecSize = MeasureSpec.getMode(heightMeasureSpec);
if (widthSpecMode == MeasureSpec.AT_MOST && heightSpecMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(width,height);
}else if (widthSpecMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(width,heightSpecSize);
}else if (heightSpecMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(widthSpecSize,height);
}
}
}
所需要的参数
<declare-styleable name="LightView">
<attr name="light_color" format="color" />
<attr name="light_radius" format="float" />
<attr name="light_src" format="float" />
</declare-styleable>
参数说明:
light_color 外发光的颜色
light_radius 外发光的范围(发散程度)
light_src 图标(与 imageview 的 src 所填内容一样)
用法 xml
<cn.guo.demo.LightView
android:layout_width="50dp"
android:layout_height="50dp"
app:light_color="#fff"
app:light_radius="5"
app:light_src="@mipmap/files_back"
android:layout_centerInParent="true"/>
效果
PS:目前只对图片有效,对 Shape.xml 无效.