Android自定义控件实现icon+文字的多种效果

今天给大家带来一个很简单但是很常用的控件ButtonExtendM,在开发中我们经常会用到图片加文字的组合控件,像这样:

以上图片都是从微信上截取的。(暂时没有找到icon在下,文字在上的例子)

下面我们通过一个控件来实现上下左右全部的样式,只需改动一个属性值即可改变icon的位置,是不是很方便,先看下demo效果图:

没错上图的三种不同的样式都是通过同一个控件实现的,下面我们看下代码

第一步 自定义属性 

在res/values/目录下新建attrs.xml文件,
添加如下属性

<attr name="backColor" format="color" />
 <attr name="backColorPress" format="color" />
 <attr name="textColor" format="color" />
 <attr name="textColorPress" format="color" />

 <declare-styleable name="ButtonExtendM">
  <attr name="backColor"/>
  <attr name="backColorPress"/>
  <attr name="textColor"/>
  <attr name="textColorPress"/>
  <attr name="iconDrawable" format="reference" />
  <attr name="iconDrawablePress" format="reference" />
  <attr name="text" format="string" />
  <attr name="textSize" format="float" />
  <attr name="spacing" format="dimension" />
  <attr name="style">
   <enum name="iconLeft" value="0" />
   <enum name="iconRight" value="1" />
   <enum name="iconUp" value="2" />
   <enum name="iconBottom" value="3" />
  </attr>
 </declare-styleable>

第二步 新建布局文件view_button_extend_m.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content">

 <ImageView
  android:id="@+id/iv_icon"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"/>

 <TextView
  android:id="@+id/tv_content"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:visibility="gone"
  android:text="@string/button_extend_m_default_text"/>

</RelativeLayout>

第三步 新建ButtonExtendM.java继承RelativeLayout

package com.landptf.view;

import android.content.Context;
import android.content.res.ColorStateList;
import android.content.res.TypedArray;
import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.widget.ImageView;
import android.widget.RelativeLayout;
import android.widget.TextView;

import com.landptf.R;
import com.landptf.util.ConvertM;

/**
 * Created by landptf on 2016/10/31.
 * 扩展Button,支持文字和icon分上下左右四种方式显示
 * 默认为左右结构,图片在左,文字在右
 */
public class ButtonExtendM extends RelativeLayout {
 /**
  * 左右结构,图片在左,文字在右
  */
 public static final int STYLE_ICON_LEFT = 0;
 /**
  * 左右结构,图片在右,文字在左
  */
 public static final int STYLE_ICON_RIGHT = 1;
 /**
  * 上下结构,图片在上,文字在下
  */
 public static final int STYLE_ICON_UP = 2;
 /**
  * 上下结构,图片在下,文字在上
  */
 public static final int STYLE_ICON_DOWN = 3;

 /**
  * 定义控件
  */
 private ImageView ivIcon;
 private TextView tvContent;
 /**
  * 上下文
  */
 private Context mContext;
 /**
  * View的背景色
  */
 private int backColor = 0;
 /**
  * View被按下时的背景色
  */
 private int backColorPress = 0;
 /**
  * icon的背景图片
  */
 private Drawable iconDrawable = null;
 /**
  * icon被按下时显示的背景图片
  */
 private Drawable iconDrawablePress = null;
 /**
  * View文字的颜色
  */
 private ColorStateList textColor = null;
 /**
  * View被按下时文字的颜色
  */
 private ColorStateList textColorPress = null;
 /**
  * 两个控件之间的间距,默认为8dp
  */
 private int spacing = 8;
 /**
  * 两个控件的位置结构
  */
 private int mStyle = STYLE_ICON_LEFT;
 /**
  * 标示onTouch方法的返回值,用来解决onClick和onTouch冲突问题
  */
 private boolean isCost = true;

 private OnClickListener onClickListener = null;

 public interface OnClickListener {
  void onClick(View v);
 }

 /**
  * 设置View的Click事件
  *
  * @param l
  */
 public void setOnClickListener(OnClickListener l) {
  this.onClickListener = l;
  isCost = false;
 }

 public ButtonExtendM(Context context) {
  super(context);
  mContext = context;
 }

 public ButtonExtendM(Context context, AttributeSet attrs) {
  this(context, attrs, 0);
 }

 public ButtonExtendM(Context context, AttributeSet attrs, int defStyle) {
  super(context, attrs, defStyle);
  mContext = context;
  init(context, attrs, defStyle);

 }

 private void init(Context context, AttributeSet attrs, int defStyle) {
  //加载布局
  LayoutInflater.from(context).inflate(R.layout.view_button_extend_m, this, true);
  //初始化控件
  ivIcon = (ImageView) findViewById(R.id.iv_icon);
  tvContent = (TextView) findViewById(R.id.tv_content);
  setGravity(Gravity.CENTER);
  TypedArray a = getContext().obtainStyledAttributes(
    attrs, R.styleable.ButtonExtendM, defStyle, 0);
  if (a != null) {
   //设置背景色
   ColorStateList colorList = a.getColorStateList(R.styleable.ButtonExtendM_backColor);
   if (colorList != null) {
    backColor = colorList.getColorForState(getDrawableState(), 0);
    if (backColor != 0) {
     setBackgroundColor(backColor);
    }
   }
   //记录View被按下时的背景色
   ColorStateList colorListPress = a.getColorStateList(R.styleable.ButtonExtendM_backColorPress);
   if (colorListPress != null) {
    backColorPress = colorListPress.getColorForState(getDrawableState(), 0);
   }
   //设置icon
   iconDrawable = a.getDrawable(R.styleable.ButtonExtendM_iconDrawable);
   if (iconDrawable != null) {
    ivIcon.setImageDrawable(iconDrawable);
   }
   //记录View被按下时的icon的图片
   iconDrawablePress = a.getDrawable(R.styleable.ButtonExtendM_iconDrawablePress);
   //设置文字的颜色
   textColor = a.getColorStateList(R.styleable.ButtonExtendM_textColor);
   if (textColor != null) {
    tvContent.setTextColor(textColor);
   }
   //记录View被按下时文字的颜色
   textColorPress = a.getColorStateList(R.styleable.ButtonExtendM_textColorPress);
   //设置显示的文本内容
   String text = a.getString(R.styleable.ButtonExtendM_text);
   if (text != null) {
    //默认为隐藏的,设置文字后显示出来
    tvContent.setVisibility(VISIBLE);
    tvContent.setText(text);
   }
   //设置文本字体大小
   float textSize = a.getFloat(R.styleable.ButtonExtendM_textSize, 0);
   if (textSize != 0) {
    tvContent.setTextSize(textSize);
   }
   //设置两个控件之间的间距
   spacing = a.getDimensionPixelSize(R.styleable.ButtonExtendM_spacing, ConvertM.dp2px(context, 8));
   //设置两个控件的位置结构
   mStyle = a.getInt(R.styleable.ButtonExtendM_style, 0);
   setIconStyle(mStyle);
   a.recycle();
  }

  setOnTouchListener(new OnTouchListener() {
   @Override
   public boolean onTouch(View arg0, MotionEvent event) {
    //根据touch事件设置按下抬起的样式
    return setTouchStyle(event.getAction());
   }
  });

  setOnClickListener(new View.OnClickListener() {
   @Override
   public void onClick(View v) {
    if (onClickListener != null) {
     onClickListener.onClick(v);
    }
   }
  });
 }

 /**
  * 根据按下或者抬起来改变背景和文字样式
  *
  * @param state
  * @return isCost
  */
 private boolean setTouchStyle(int state) {
  if (state == MotionEvent.ACTION_DOWN) {
   if (backColorPress != 0) {
    setBackgroundColor(backColorPress);
   }
   if (iconDrawablePress != null) {
    ivIcon.setImageDrawable(iconDrawablePress);
   }
   if (textColorPress != null) {
    tvContent.setTextColor(textColorPress);
   }
  }
  if (state == MotionEvent.ACTION_UP) {
   if (backColor != 0) {
    setBackgroundColor(backColor);
   }
   if (iconDrawable != null) {
    ivIcon.setImageDrawable(iconDrawable);
   }
   if (textColor != null) {
    tvContent.setTextColor(textColor);
   }
  }
  return isCost;
 }

 /**
  * 设置图标位置
  * 通过重置LayoutParams来设置两个控件的摆放位置
  * @param style
  */
 public void setIconStyle(int style) {
  mStyle = style;
  RelativeLayout.LayoutParams lp;
  switch (style) {
   case STYLE_ICON_LEFT:
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_VERTICAL);
    ivIcon.setLayoutParams(lp);
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_VERTICAL);
    lp.addRule(RelativeLayout.RIGHT_OF, ivIcon.getId());
    lp.leftMargin = spacing;
    tvContent.setLayoutParams(lp);
    break;
   case STYLE_ICON_RIGHT:
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_VERTICAL);
    tvContent.setLayoutParams(lp);
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_VERTICAL);
    lp.addRule(RelativeLayout.RIGHT_OF, tvContent.getId());
    lp.leftMargin = spacing;
    ivIcon.setLayoutParams(lp);
    break;
   case STYLE_ICON_UP:
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_HORIZONTAL);
    ivIcon.setLayoutParams(lp);
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_HORIZONTAL);
    lp.addRule(RelativeLayout.BELOW, ivIcon.getId());
    lp.leftMargin = spacing;
    tvContent.setLayoutParams(lp);
    break;
   case STYLE_ICON_DOWN:
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_HORIZONTAL);
    tvContent.setLayoutParams(lp);
    lp = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
    lp.addRule(RelativeLayout.CENTER_HORIZONTAL);
    lp.addRule(RelativeLayout.BELOW, tvContent.getId());
    lp.leftMargin = spacing;
    ivIcon.setLayoutParams(lp);
    break;
   default:
    break;
  }
 }

 /**
  * 设置View的背景色
  *
  * @param backColor
  */
 public void setBackColor(int backColor) {
  this.backColor = backColor;
  setBackgroundColor(backColor);
 }

 /**
  * 设置View被按下时的背景色
  *
  * @param backColorPress
  */
 public void setBackColorPress(int backColorPress) {
  this.backColorPress = backColorPress;
 }

 /**
  * 设置icon的图片
  *
  * @param iconDrawable
  */
 public void setIconDrawable(Drawable iconDrawable) {
  this.iconDrawable = iconDrawable;
  ivIcon.setImageDrawable(iconDrawable);
 }

 /**
  * 设置View被按下时的icon的图片
  *
  * @param iconDrawablePress
  */
 public void setIconDrawablePress(Drawable iconDrawablePress) {
  this.iconDrawablePress = iconDrawablePress;
 }

 /**
  * 设置文字的颜色
  *
  * @param textColor
  */
 public void setTextColor(int textColor) {
  if (textColor == 0) return;
  this.textColor = ColorStateList.valueOf(textColor);
  tvContent.setTextColor(this.textColor);
 }

 /**
  * 设置View被按下时文字的颜色
  *
  * @param textColorPress
  */
 public void setTextColorPress(int textColorPress) {
  if (textColorPress == 0) return;
  this.textColorPress = ColorStateList.valueOf(textColorPress);
 }

 /**
  * 设置显示的文本内容
  *
  * @param text
  */
 public void setText(CharSequence text) {
  //默认为隐藏的,设置文字后显示出来
  tvContent.setVisibility(VISIBLE);
  tvContent.setText(text);
 }

 /**
  * 获取显示的文本
  *
  * @return
  */
 public String getText() {
  return tvContent.getText().toString();
 }

 /**
  * 设置文本字体大小
  *
  * @param size
  */
 public void setTextSize(float size) {
  tvContent.setTextSize(size);
 }

 /**
  * 设置两个控件之间的间距
  *
  * @param spacing
  */
 public void setSpacing(int spacing) {
  this.spacing = ConvertM.dp2px(mContext, spacing);
  //设置完成后刷新一下两个控件的结构,避免先执行了setIconStyle后,setSpacing不生效
  setIconStyle(mStyle);
 }

}

代码注释基本可以看懂具体的实现,接下来主要看下如何使用

在layout里直接引用ButtonExtendM即可,注意要添加

xmlns:landptf="http://schemas.android.com/apk/res-auto"

<com.landptf.view.ButtonExtendM
 android:id="@+id/bem_back"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_centerVertical="true"
 android:layout_marginLeft="8dp"
 landptf:iconDrawable="@drawable/title_back"
 landptf:iconDrawablePress="@drawable/title_back_selected"
 landptf:textColor="@android:color/white"
 landptf:spacing="4dp"
 landptf:text="返回"/>

这个是实现的菜单栏的返回按钮,左右结构,icon在左为默认样式,注意一下*Press的属性,主要是用来设置控件被按下后的效果的。

再来看一个上下结构的

<com.landptf.view.ButtonExtendM
 android:layout_width="0dp"
 android:layout_height="match_parent"
 android:layout_weight="1"
 landptf:iconDrawable="@drawable/icon_home_page"
 landptf:text="首页"
 landptf:style="iconUp" />

只需要设置landptf:style即可,同时也可以通过java代码实现

setIconStyle(ButtonExtendM.STYLE_ICON_UP)

全部代码已托管到开源中国的码云上,欢迎下载,地址:https://git.oschina.net/landptf/landptf.git

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • android虚拟键盘弹出遮挡登陆按钮问题的解决方法

    Android虚拟键盘的弹起会遮挡住部分ui,虽然通过在清单文件中设置,可以随着虚拟键盘的弹出,布局往上推,但是面对登陆界面时,并没有太大的作用,这样就会导致用户体验不好:开发中既然出现了就的解决:先说先解决的思路:获取到屏幕的高度.虚拟键盘的高度,布局的高度,用屏幕的高度减去布局的高度,用高度差和虚拟键盘的高度进行对比:代码实现如下: private LinearLayout logo_layout; private ImageView iv_logo; private int sh; pri

  • Android 获取手机信息实例详解

    Android 获取手机信息 应用信息:包名.版本号.版本名,手机是否有Root权限 手机信息:手机屏幕宽和高.当前可用内存大小.总内存大小.IMEI号.IESI号.手机型号.手机品牌.手机MacAdd.CPU型号.CPU频率 开门见山,以下是Java代码,XML只有一个TextView显示信息. package com.example.getphoneinfo; import java.io.BufferedReader; import java.io.File; import java.io

  • Android自定义控件实现底部菜单(下)

    在app中经常会用到底部菜单的控件,每次都需要写好多代码,今天我们用到了前几篇博客里的控件来进一步封装底部菜单.先看效果图: 主要包括以下功能: 1 设置icon以及点击之后的icon 2 设置文字 3 设置文字颜色以及点击之后的文字颜色 4 设置未读数量.更多以及new 我们先看如何使用,然后再看如何实现的 1 在布局文件中引用MenuM <com.landptf.view.MenuM android:id="@+id/mm_bottom" android:layout_wid

  • Android编程实现自定义ProgressBar样式示例(背景色及一级、二级进度条颜色)

    本文实例讲述了Android编程实现自定义ProgressBar样式.分享给大家供大家参考,具体如下: 效果图如下,本例中设置了第一级进度条和第二级进度条. 样式资源:progressbar_bg.xml,放在drawable文件夹下: <?xml version="1.0" encoding="utf-8"?> <layer-list xmlns:android="http://schemas.android.com/apk/res/a

  • Android Secret Code(输入字符弹出手机信息)详解

    Android Secret Code 我们很多人应该都做过这样的操作,打开拨号键盘输入*#*#4636#*#*等字符就会弹出一个界面显示手机相关的一些信息,这个功能在Android中被称为android secret code,除了这些系统预置的secret code,我们也可以实现自己的secret code,而且实现起来非常简单. 要实现自己的secret code,只需要向系统注册一个Broadcast Receiver,不需要任何权限,如下所示: <receiver android:n

  • Android编程获取设备MAC地址的实现方法

    本文实例讲述了Android编程获取设备MAC地址的实现方法.分享给大家供大家参考,具体如下: /** * 获取设备的mac地址 * * @param ac * @param callback * 成功获取到mac地址之后会回调此方法 */ public static void getMacAddress(final Activity ac, final SimpleCallback callback) { final WifiManager wm = (WifiManager) ac .get

  • Android BadgeView红点更新信息提示示例代码

    应用市场很多应用程序中都会看见一些数字红点提示的效果,如QQ.微信以及一些提示更新应用的APP,以达到更好的提示功能的应用,本文将介绍一开源控件的使用实现红点更新信息提示效果. 一.BadgeView常用方法介绍: 1.setBadgeCount(int):设置提醒数字 2.setBadgeGravity(Gravity):设置位置布局 3.setTargetView(View):设置提示控件对象 4.setTypeface():设置显示字体 5.setShadowLayer():设置字体阴影

  • Android中获取资源 id 及资源 id 的动态获取

     Android中获取资源 id 及资源 id 的动态获取 我们平时获取资源是通过 findViewById 方法进行的,比如我们常在onCreate方法中使用这样的语句: btnChecked=(ImageView)findViewById(R.id.imgCheck); findViewById是我们获取layout中各种View 对象比如按钮.标签.ListView和ImageView的便利方法.顾名思义,它需要一个int参数:资源id. 资源id非常有用.Android回自动为每个位于r

  • Android自定义控件实现icon+文字的多种效果

    今天给大家带来一个很简单但是很常用的控件ButtonExtendM,在开发中我们经常会用到图片加文字的组合控件,像这样: 以上图片都是从微信上截取的.(暂时没有找到icon在下,文字在上的例子) 下面我们通过一个控件来实现上下左右全部的样式,只需改动一个属性值即可改变icon的位置,是不是很方便,先看下demo效果图: 没错上图的三种不同的样式都是通过同一个控件实现的,下面我们看下代码 第一步 自定义属性  在res/values/目录下新建attrs.xml文件, 添加如下属性 <attr n

  • Android 自定义控件实现显示文字的功能

    Android 自定义控件实现显示文字的功能 自定义控件-–逐个显示文字 ONE Goal ,ONE Passion ! 前言: 今天要实现的效果时.让我们的文字一个一个显示出来.上效果图吧: 实现原理: 1,拿到要显示的文字. 2,计算文字显示的速率 字体显示的速度 v = 总的字体长度 / 总的显示时间 3,将文字根据速率显示到控件上. 自定义View: public class printTextView extends TextView { /** * 字体显示出来的时间 */ priv

  • Android自定义控件仿ios下拉回弹效果

    网上有很多类似的文章,大多数还是继承listview来实现(主要是listview.addHeaderView()和listview.addFooterView在listview的首尾添加view,也可以用上面的两个listview自带函数实现下拉刷新的功能,在这里不准备介绍,有兴趣的朋友可以去自己试试). 在本文主要是给android的线性布局(相对布局.帧布局)加上下拉或者上拉回弹得效果.在ios中我们经常能看到,在一个页面中即使是只有一个控件,这一个控件只占整个页面的1/10不到,但是当我

  • Android自定义控件实现带文字提示的SeekBar

    1.写在前面 SeekBar控件在开发中还是比较常见的,比如音视频进度.音量调节等,但是原生控件有时还不能满足我们的需求,今天就来学习一下如何自定义SeekBar控件,本文主要实现了一个带文字指示器效果的SeekBar控件 看下最终效果: IndicatorSeekBar 2.实现 IndicatorSeekBar public class IndicatorSeekBar extends AppCompatSeekBar { // 画笔 private Paint mPaint; // 进度文

  • Android自定义控件实现UC浏览器语音搜索效果

    最近项目上要实现语音搜索功能,界面样式要模仿一下UC浏览器的样式,UC浏览器中有一个控件,会随着声音大小浮动,然后寻思偷个懒,百度一下,结果也没有找到类似的,只能自己动手了. 先上图看我实现的效果: 这是自定义控件的代码,里面注释也很明白,就不费话了 public class CustomCircleView extends View{ private Paint mPaint; private int strokeWidth = 0; //圆环的宽度 private Bitmap bitmap

  • Android自定义控件实现边缘凹凸的卡劵效果

    前言 最近做项目的时候遇到一个卡劵的效果,由于自己觉得用图片来做的话可以会出现适配效果不好,再加上自己自定义view方面的知识比较薄弱,所以想试试用自定义View来实现.但是由于自己知识点薄弱,一开始居然想着用画矩形来设置边缘实现,后面一个哥们指导了我,在这里感谢他. 实现分析 上面的图片其实和普通的Linearlayout,RelativeLayout一样,只是上下两边多了类似于半圆锯齿的形状.那么只需要处理不同地方.可以在上下两条线上画一个个白色的小圆来实现这种效果. 假如我们上下线的半圆以

  • Android自定义控件实现滑动开关效果

    自定义开关控件 Android自定义控件一般有三种方式 1.继承Android固有的控件,在Android原生控件的基础上,进行添加功能和逻辑. 2.继承ViewGroup,这类自定义控件是可以往自己的布局里面添加其他的子控件的. 3.继承View,这类自定义控件没有跟原生的控件有太多的相似的地方,也不需要在自己的肚子里添加其他的子控件. ToggleView自定义开关控件表征上没有跟Android原生的控件有什么相似的地方,而且在滑动的效果上也没有沿袭Android原生的地方,所以我们的自定义

  • Android 实现带头部文字输入框的自定义控件

    前言 在app的输入框中,需要应用到很多带有前缀说明的输入框,运用原有的输入框和文本控件,一个带头部的输入框就会增加三个控件在layout文件中.当布局文件输入框较少的情况下,这样对后期维护影响不大,但在多个带头部的输入框下,布局文件代码量会很大,影响阅读以及后期维护.而封装过后的控件,在使用中仅仅需要几行代码可实现几十行的效果. 简介 带头部文字的输入框 可在xml定义头部文字样式 可在xml定义输入框样式 可在xml定义提示文字样式 可在xml定义头部和输入框的间距和边距 效果图 使用方法

  • Android自定义控件eBook实现翻书效果实例详解

    本文实例讲述了Android自定义控件eBook实现翻书效果的方法.分享给大家供大家参考,具体如下: 效果图: Book.java文件: package com.book; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.ImageView; public class Book extend

  • Android中使用TextView实现文字跑马灯效果

    通常情况下我们想实现文字的走马灯效果需要在xml文件中这样设置 <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:singleLine="true" android:ellipsize="marquee" android:focusable="true" android:

随机推荐