原Github地址:https://github.com/hoang8f/android-flat-button
使用方法:
1.加入gradle依赖:
compile 'info.hoang8f:fbutton:1.0.5'
2.加入FButton的java文件:
import android.content.Context;
import android.content.res.Resources;
import android.content.res.TypedArray;
import android.graphics.Color;
import android.graphics.Rect;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.LayerDrawable;
import android.graphics.drawable.ShapeDrawable;
import android.graphics.drawable.shapes.RoundRectShape;
import android.os.Build;
import android.support.v7.widget.AppCompatButton;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import info.hoang8f.fbutton.R;
/**
* Created by hoang8f on 5/5/14.
*/
public class FButton extends AppCompatButton implements View.OnTouchListener {
//Custom values
private boolean isShadowEnabled = true;
private int mButtonColor;
private int mShadowColor;
private int mShadowHeight;
private int mCornerRadius;
//Native values
private int mPaddingLeft;
private int mPaddingRight;
private int mPaddingTop;
private int mPaddingBottom;
//Background drawable
private Drawable pressedDrawable;
private Drawable unpressedDrawable;
boolean isShadowColorDefined = false;
public FButton(Context context) {
super(context);
init();
this.setOnTouchListener(this);
}
public FButton(Context context, AttributeSet attrs) {
super(context, attrs);
init();
parseAttrs(context, attrs);
this.setOnTouchListener(this);
}
public FButton(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init();
parseAttrs(context, attrs);
this.setOnTouchListener(this);
}
@Override
protected void onFinishInflate() {
super.onFinishInflate();
//Update background color
refresh();
}
@Override
public boolean onTouch(View view, MotionEvent motionEvent) {
switch (motionEvent.getAction()) {
case MotionEvent.ACTION_DOWN:
updateBackground(pressedDrawable);
this.setPadding(mPaddingLeft, mPaddingTop + mShadowHeight, mPaddingRight, mPaddingBottom);
break;
case MotionEvent.ACTION_MOVE:
Rect r = new Rect();
view.getLocalVisibleRect(r);
if (!r.contains((int) motionEvent.getX(), (int) motionEvent.getY() + 3 * mShadowHeight) &&
!r.contains((int) motionEvent.getX(), (int) motionEvent.getY() - 3 * mShadowHeight)) {
updateBackground(unpressedDrawable);
this.setPadding(mPaddingLeft, mPaddingTop + mShadowHeight, mPaddingRight, mPaddingBottom + mShadowHeight);
}
break;
case MotionEvent.ACTION_OUTSIDE:
case MotionEvent.ACTION_CANCEL:
case MotionEvent.ACTION_UP:
updateBackground(unpressedDrawable);
this.setPadding(mPaddingLeft, mPaddingTop + mShadowHeight, mPaddingRight, mPaddingBottom + mShadowHeight);
break;
}
return false;
}
private void init() {
//Init default values
isShadowEnabled = true;
Resources resources = getResources();
if (resources == null) return;
mButtonColor = resources.getColor(R.color.fbutton_default_color);
mShadowColor = resources.getColor(R.color.fbutton_default_shadow_color);
mShadowHeight = resources.getDimensionPixelSize(R.dimen.fbutton_default_shadow_height);
mCornerRadius = resources.getDimensionPixelSize(R.dimen.fbutton_default_conner_radius);
}
private void parseAttrs(Context context, AttributeSet attrs) {
//Load from custom attributes
TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.FButton);
if (typedArray == null) return;
for (int i = 0; i < typedArray.getIndexCount(); i++) {
int attr = typedArray.getIndex(i);
if (attr == R.styleable.FButton_shadowEnabled) {
isShadowEnabled = typedArray.getBoolean(attr, true); //Default is true
} else if (attr == R.styleable.FButton_buttonColor) {
mButtonColor = typedArray.getColor(attr, R.color.fbutton_default_color);
} else if (attr == R.styleable.FButton_shadowColor) {
mShadowColor = typedArray.getColor(attr, R.color.fbutton_default_shadow_color);
isShadowColorDefined = true;
} else if (attr == R.styleable.FButton_shadowHeight) {
mShadowHeight = typedArray.getDimensionPixelSize(attr, R.dimen.fbutton_default_shadow_height);
} else if (attr == R.styleable.FButton_cornerRadius) {
mCornerRadius = typedArray.getDimensionPixelSize(attr, R.dimen.fbutton_default_conner_radius);
}
}
typedArray.recycle();
//Get paddingLeft, paddingRight
int[] attrsArray = new int[]{
android.R.attr.paddingLeft, // 0
android.R.attr.paddingRight, // 1
};
TypedArray ta = context.obtainStyledAttributes(attrs, attrsArray);
if (ta == null) return;
mPaddingLeft = ta.getDimensionPixelSize(0, 0);
mPaddingRight = ta.getDimensionPixelSize(1, 0);
ta.recycle();
//Get paddingTop, paddingBottom
int[] attrsArray2 = new int[]{
android.R.attr.paddingTop, // 0
android.R.attr.paddingBottom,// 1
};
TypedArray ta1 = context.obtainStyledAttributes(attrs, attrsArray2);
if (ta1 == null) return;
mPaddingTop = ta1.getDimensionPixelSize(0, 0);
mPaddingBottom = ta1.getDimensionPixelSize(1, 0);
ta1.recycle();
}
public void refresh() {
int alpha = Color.alpha(mButtonColor);
float[] hsv = new float[3];
Color.colorToHSV(mButtonColor, hsv);
hsv[2] *= 0.8f; // value component
//if shadow color was not defined, generate shadow color = 80% brightness
if (!isShadowColorDefined) {
mShadowColor = Color.HSVToColor(alpha, hsv);
}
//Create pressed background and unpressed background drawables
if (this.isEnabled()) {
if (isShadowEnabled) {
pressedDrawable = createDrawable(mCornerRadius, Color.TRANSPARENT, mButtonColor);
unpressedDrawable = createDrawable(mCornerRadius, mButtonColor, mShadowColor);
} else {
mShadowHeight = 0;
pressedDrawable = createDrawable(mCornerRadius, mShadowColor, Color.TRANSPARENT);
unpressedDrawable = createDrawable(mCornerRadius, mButtonColor, Color.TRANSPARENT);
}
} else {
Color.colorToHSV(mButtonColor, hsv);
hsv[1] *= 0.25f; // saturation component
int disabledColor = mShadowColor = Color.HSVToColor(alpha, hsv);
// Disabled button does not have shadow
pressedDrawable = createDrawable(mCornerRadius, disabledColor, Color.TRANSPARENT);
unpressedDrawable = createDrawable(mCornerRadius, disabledColor, Color.TRANSPARENT);
}
updateBackground(unpressedDrawable);
//Set padding
this.setPadding(mPaddingLeft, mPaddingTop + mShadowHeight, mPaddingRight, mPaddingBottom + mShadowHeight);
}
private void updateBackground(Drawable background) {
if (background == null) return;
//Set button background
if (Build.VERSION.SDK_INT >= 16) {
this.setBackground(background);
} else {
this.setBackgroundDrawable(background);
}
}
private LayerDrawable createDrawable(int radius, int topColor, int bottomColor) {
float[] outerRadius = new float[]{radius, radius, radius, radius, radius, radius, radius, radius};
//Top
RoundRectShape topRoundRect = new RoundRectShape(outerRadius, null, null);
ShapeDrawable topShapeDrawable = new ShapeDrawable(topRoundRect);
topShapeDrawable.getPaint().setColor(topColor);
//Bottom
RoundRectShape roundRectShape = new RoundRectShape(outerRadius, null, null);
ShapeDrawable bottomShapeDrawable = new ShapeDrawable(roundRectShape);
bottomShapeDrawable.getPaint().setColor(bottomColor);
//Create array
Drawable[] drawArray = {bottomShapeDrawable, topShapeDrawable};
LayerDrawable layerDrawable = new LayerDrawable(drawArray);
//Set shadow height
if (isShadowEnabled && topColor != Color.TRANSPARENT) {
//unpressed drawable
layerDrawable.setLayerInset(0, 0, 0, 0, 0); /*index, left, top, right, bottom*/
} else {
//pressed drawable
layerDrawable.setLayerInset(0, 0, mShadowHeight, 0, 0); /*index, left, top, right, bottom*/
}
layerDrawable.setLayerInset(1, 0, 0, 0, mShadowHeight); /*index, left, top, right, bottom*/
return layerDrawable;
}
//Setter
public void setShadowEnabled(boolean isShadowEnabled) {
this.isShadowEnabled = isShadowEnabled;
setShadowHeight(0);
refresh();
}
public void setButtonColor(int buttonColor) {
this.mButtonColor = buttonColor;
refresh();
}
public void setShadowColor(int shadowColor) {
this.mShadowColor = shadowColor;
isShadowColorDefined = true;
refresh();
}
public void setShadowHeight(int shadowHeight) {
this.mShadowHeight = shadowHeight;
refresh();
}
public void setCornerRadius(int cornerRadius) {
this.mCornerRadius = cornerRadius;
refresh();
}
public void setFButtonPadding(int left, int top, int right, int bottom) {
mPaddingLeft = left;
mPaddingRight = right;
mPaddingTop = top;
mPaddingBottom = bottom;
refresh();
}
@Override
public void setEnabled(boolean enabled) {
super.setEnabled(enabled);
refresh();
}
//Getter
public boolean isShadowEnabled() {
return isShadowEnabled;
}
public int getButtonColor() {
return mButtonColor;
}
public int getShadowColor() {
return mShadowColor;
}
public int getShadowHeight() {
return mShadowHeight;
}
public int getCornerRadius() {
return mCornerRadius;
}
}
3.创建attrs.xml
4.创建 color.xml
#3F51B5
#303F9F
#FF4081
#3eadeb
#ff3493c8
//Color Swatches provided by http://designmodo.github.io/Flat-UI/
#1abc9c
#16a085
#2ecc71
#27ae60
#3498db
#2980b9
#9b59b6
#8e44ad
#34495e
#2c3e50
#f1c40f
#f39c12
#e67e22
#d35400
#e74c3c
#c0392b
#ecf0f1
#bdc3c7
#95a5a6
#7f8c8d
@android:color/transparent
5.创建 dimens.xml
8dp
10dp
10dp
5dp
5dp
4dp
6.使用:
在layout中的根节点加入
xmlns:fbutton="http://schemas.android.com/apk/res-auto"
接着使用
注意那个com.example.administrator.button_demo,要改成自己的包