Android编程中TextView宽度过大导致Drawable无法居中问题解决方法_Android

本文实例讲述了Android编程中TextView宽度过大导致Drawable无法居中问题解决方法。分享给大家供大家参考,具体如下:

在做项目的时候,很多时候我们都要用到文字和图片一起显示,一般设置TextView的DrawableLeft、DrawableRight、DrawableTop、DrawableBottom就行了。但是有一种情况是当TextView的熟悉是fill_parent或者使用权重的时候并且设置了起Gravity的ceter的时候,Drawable图片是无法一起居中的,为了解决其,我们一般再套一层布局,然后设置TextView的熟悉是wrap_content,但是有时候嵌套过多的布局的时候,有可能发生StackOverFlow,所以必须要优化,下面说一下其中的一个解决方案。先上图

这个解决方案很粗糙,局限性很大,文字不能换行,换行之后就不准了,下面是源码:

package com.example.testandroid;
import java.lang.ref.WeakReference;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.TextView;
public class DrawableTextView extends TextView {
 private WeakReference<Bitmap> normalReference;
 private WeakReference<Bitmap> pressReference;
 private WeakReference<Bitmap> showReference;
 private int normalColor = Color.WHITE, pressColor = Color.WHITE;
 private String text;
 private int textWidth = 0;
 private int textHeight = 0;
 public DrawableTextView(Context context) {
  super(context);
 }
 public DrawableTextView(Context context, AttributeSet attrs) {
  super(context, attrs);
 }
 public DrawableTextView(Context context, AttributeSet attrs, int defStyle) {
  super(context, attrs, defStyle);
 }
 @Override
 protected void onFinishInflate() {
  super.onFinishInflate();
  initText();
 }
 private void initText() {
  text = super.getText().toString();
  initVariable();
 }
 /**
  * 初始化,测量Textview内容的长度,高度
  */
 private void initVariable() {
  textWidth = (int) (getPaint().measureText(text));
  final Rect rect = new Rect();
  getPaint().getTextBounds(text, 0, 1, rect);
  textHeight = rect.height();
 }
 /**
  * 设置TextView的内容
  * @param text
  */
 public void setText(String text) {
  this.text = text;
  initVariable();
  invalidate();
 }
 /**
  * 获取TextView内容
  */
 public String getText() {
  return text;
 }
 /**
  * 设置TextView的Drawable内容,目前仅支持DrawableLeft
  * @param normalDrawableId
  *    DrawableLeft的normal状态Id
  * @param pressDrawableId
  *    DrawableLeft的press状态的Id(没有press状态,请传-1)
  */
 public void setDrawableLeftId(final int normalDrawableId, final int pressDrawableId) {
  normalReference = new WeakReference<Bitmap>(BitmapFactory.decodeResource(getResources(), normalDrawableId));
  if (pressDrawableId != -1) {
   pressReference = new WeakReference<Bitmap>(BitmapFactory.decodeResource(getResources(), pressDrawableId));
  }
  showReference = normalReference;
  invalidate();
 }
 /**
  * 设置TextView的Color
  * @param normalColor
  *    TextView normal状态的Color值
  * @param pressDrawableId
  *    TextView press状态的Color值(如果没有press状态,请传与normal状态的值)
  */
 public void setTextColor(final int normalColor, final int pressColor) {
  this.normalColor = normalColor;
  this.pressColor = pressColor;
  getPaint().setColor(normalColor);
  initVariable();
 }
 @Override
 protected void onDraw(Canvas canvas) {
  if (showReference != null && showReference.get() != null) {
   final int bitmapWidth = showReference.get().getWidth();
   final int bitmapHeight = showReference.get().getHeight();
   final int viewHeight = getHeight();
   final int drawablePadding = getCompoundDrawablePadding();
   final int start = (getWidth() - (bitmapWidth + drawablePadding + textWidth)) >> 1;
   canvas.drawBitmap(showReference.get(), start, (viewHeight >> 1) - (bitmapHeight >> 1), getPaint());
   /**
    * 注意改方法,第三个参数y,本人也被误导了好久,原来在画文字的时候,y表示文字最后的位置(不是下笔点的起始位置)
    * 所以为什么 是TextView高度的一半(中间位置) + 文字高度的一半 = 文字居中
    */
   canvas.drawText(text, start + drawablePadding + bitmapWidth, (viewHeight >> 1) + (textHeight >> 1), getPaint());
  }
 }
 @Override
 public boolean onTouchEvent(MotionEvent event) {
  if (event.getAction() == MotionEvent.ACTION_DOWN) {
   if (pressReference != null && pressReference.get() != null) {
    showReference = pressReference;
   }
   getPaint().setColor(pressColor);
  } else if (event.getAction() == MotionEvent.ACTION_UP) {
   if (normalReference != null && normalReference.get() != null) {
    showReference = normalReference;
   }
   getPaint().setColor(normalColor);
  }
  invalidate();
  return super.onTouchEvent(event);
 }
}

xml布局:

<com.example.testandroid.DrawableTextView
android:id="@+id/my_textview"
android:layout_width="fill_parent"
android:layout_marginTop="20dp"
android:background="@drawable/text_selector"
android:drawablePadding="8dp"
android:textColor="@color/standard_orange"
android:layout_height="wrap_content"
android:padding="15dp"
android:textSize="16sp"
android:text="有Drawable的TextView" />

调用代码:

DrawableTextView drawableTextView = (DrawableTextView) getView().findViewById(R.id.my_textview);
drawableTextView.setDrawableLeftId(R.drawable.bg_btn_delete_normal, R.drawable.bg_btn_delete_pressed);
drawableTextView.setTextColor(getResources().getColor(R.color.standard_orange), getResources().getColor(R.color.standard_white));
drawableTextView.setText("我在动态修改Text啦");

其实还有更加方便的方法,下面朋友借鉴某个网友的代码(地址我就不知道了):

@Override
protected void onDraw(Canvas canvas) {
 Drawable[] drawables = getCompoundDrawables();
 if (drawables != null) {
  Drawable drawableLeft = drawables[0];
  if (drawableLeft != null) {
   final float textWidth = getPaint().measureText(getText().toString());
   final int drawablePadding = getCompoundDrawablePadding();
   final int drawableWidth = drawableLeft.getIntrinsicWidth();
   final float bodyWidth = textWidth + drawableWidth + drawablePadding;
   canvas.translate((getWidth() - bodyWidth) / 2, 0);
  }
 }
 super.onDraw(canvas);
}

xml布局:

<com.example.testandroid.DrawableTextView
android:id="@+id/my_textview"
android:layout_width="fill_parent"
android:layout_marginTop="20dp"
android:background="@drawable/text_selector"
android:drawablePadding="8dp"
android:drawableLeft="@drawable/clear_edittext_selector"
android:textColor="@color/text_color_selector"
android:layout_height="wrap_content"
android:padding="15dp"
android:textSize="16sp"
android:text="有Drawable的TextView" />

嗯,自己写这个东西,也学到了一些东西,大家有什么更好的方法,大家可以讨论一下。

希望本文所述对大家Android程序设计有所帮助。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索android
, textview
, drawable
, 解决方法
, 宽度过大
无法居中
textview 文字居中、text align 垂直居中、svg text 居中、android textview居中、textview垂直居中,以便于您获取更多的相关知识。

时间: 2024-08-01 18:01:08

Android编程中TextView宽度过大导致Drawable无法居中问题解决方法_Android的相关文章

Android关于TextView 宽度过大导致Drawable无法居中问题

在做项目的时候,很多时候我们都要用到文字和图片一起显示,一般设置TextView的DrawableLeft.DrawableRight.DrawableTop.DrawableBottom就行了.但是有一种情况是当TextView的熟悉是fill_parent或者使用权重的时候并且设置了起Gravity的ceter的时候,Drawable图片是无法一起居中的,为了解决其,我们一般再套一层布局,然后设置TextView的熟悉是wrap_content,但是有时候嵌套过多的布局的时候,有可能发生St

Android编程中TextView字体属性设置方法(大小、字体、下划线、背景色)_Android

本文实例讲述了Android编程中TextView字体属性设置方法(大小.字体.下划线.背景色).分享给大家供大家参考,具体如下: import android.content.Context; import android.graphics.Color; import android.text.SpannableString; import android.text.Spanned; import android.text.style.AbsoluteSizeSpan; import andr

Android开发中TextView 实现右上角跟随文本动态追加圆形红点_Android

在一个比较坑的需求里,一段文字右上角需要追加一个圆形红点.最右侧有个金额,红点动态随着文字移动,然后各种摆布局,一下午坑死我了.后来果断放弃.然后就想试试直接自定义view来实现这个需求. 最坑的就是效果下面的第一种情况和第二种情况,就是这两种情况给逼的   废话不说,开搞. 首先自定义个view 继承自 view 类 public class MyViewAndCircle extends View{ } 然后不用说了 ,直接飘红,必须要实现几个必要的方法了. public MyViewAnd

Android编程实现ViewPager多页面滑动切换及动画效果的方法_Android

本文实例讲述了Android编程实现ViewPager多页面滑动切换及动画效果的方法.分享给大家供大家参考,具体如下: 一.首先,我们来看一下效果图,这是新浪微博的Tab滑动效果.我们可以手势滑动,也可以点击上面的头标进行切换.与此同方式, 白色横条会移动到相应的页卡头标下.这是一个动画效果,白条是缓慢滑动过去的.好了,接下来我们就来实现它. 二.在开始前,我们先要认识一个控件,ViewPager.它是google SDk中自带的一个附加包的一个类,可以用来实现屏幕间的切换. 这个附加包是and

Android编程实现基于局域网udp广播自动建立socket连接的方法_Android

本文实例讲述了Android编程实现基于局域网udp广播自动建立socket连接的方法.分享给大家供大家参考,具体如下: android开发中经常会用到socket通讯.由于项目需要,最近研究了一下这方面的知识. 需求是想通过wifi实现android移动设备和android平台的电视之间的文件传输与控制. 毫无疑问这中间一定需要用到socket来进行通信.今天就两台设备的握手连接方式分享一下吧,该方法只是本人个人想法的实现,仅供参考,如有雷同,不胜荣幸. 要想使用socket进行通讯,就必须知

Android Activity中使用Intent实现页面跳转与参数传递的方法_Android

本文实例讲述了Android Activity中使用Intent实现页面跳转与参数传递的方法.分享给大家供大家参考,具体如下: 新建一个FirstAvtivity.java package com.zhuguangwei; import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.view.View; import android.view.View.O

Android编程简单实现ImageView点击时背景图修改的方法_Android

本文实例讲述了Android编程简单实现ImageView点击时背景图修改的方法.分享给大家供大家参考,具体如下: 在使用ImageView时,当被点击时,希望背景图修改一下,这样显示被点击效果明显一些.在这里,一个很简单的方法,最起码是个很清晰的方法.在res/drawable文件夹下创建一个xml文件.比如my.xml,内容如下: <?xml version="1.0" encoding="utf-8"?> <selector xmlns:an

Android viewpager中动态添加view并实现伪无限循环的方法_Android

本文实例讲述了Android viewpager中动态添加view并实现伪无限循环的方法.分享给大家供大家参考,具体如下: viewpager的使用,大家都熟悉,它可以实现页面之间左右滑动的切换,这里有一个需求,就是viewpager里面加载的页数不是确定的,而是根据数据的多少来确定的.常见的应用就是在一个新闻的详细页面中,显示与此新闻有关的图片. 下面我们来看一下代码: activity_main.xml <RelativeLayout xmlns:android="http://sch

Android编程使WebView支持HTML5 Video全屏播放的解决方法_Android

本文实例讲述了Android编程使WebView支持HTML5 Video全屏播放的解决方法.分享给大家供大家参考,具体如下: 1)需要在AndroidManifest.xml文件中声明需要使用HardwareAccelerate, 可以细化到Activity级别,如果不需要的View可以声明不要用加速,但是需要在代码中做,具体如下: a. 如果要声明整个应用都要加速: 复制代码 代码如下: <application ... android:hardwareAccelerated ="tr