在项目中,回复框、聊天界面的显示往往会有emoj或者图片,但是一个比较头疼的问题是,会出现emoj表情或者图片和文字的位置不对齐,总是有偏移,这样很影响用户体验的。下面会总结一下如何解决这个问题。

本文所列举的解决方案是参考一个非常给力的表情键盘项目:https://github.com/w446108264/XhsEmoticonsKeyboard

另外一个非常给力的是google提供的开源demo:https://github.com/googlesamples/android-EmojiCompat,文档地址为:https://developer.android.com/guide/topics/ui/look-and-feel/emoji-compat.html#using-emojicompat-for-imes

1. TextView设置

/**
* 解析评论内容,包含富文本表情
*
* @param textView
* @param text
*/
private void setRichText(TextView textView, String text) {
// 将文本转换为span
Spanned span = Html.fromHtml(text);
int fontSize = UIHelper.getFontHeight(textView);
if (TextUtils.isEmpty(text)) {
textView.setVisibility(View.GONE);
} else {
textView.setFocusable(false);
textView.setLongClickable(false);
span = InputHelper.displayEmoji(context.getResources(), span, fontSize, Math.round(TDevice.dp2px(5.0f/2.0f)) );
// 去除下划线
SpannableString spannableString = new SpannableString(span);
spannableString.setSpan(new NoUnderlineSpan(),,span.length(),Spanned.SPAN_MARK_MARK);
textView.setText(spannableString);
}
}

2. 展示文字和表情

/**
* 展示文字和表情
* @param res
* @param s
* @param fontSize 字体大小
* @param lineSpace 行间距
* @return
*/
public static Spannable displayEmoji(Resources res, CharSequence s, int fontSize, int lineSpace) {
String str = s.toString();
Spannable spannable;
if (s instanceof Spannable) {
spannable = (Spannable) s;
} else {
// 构建文字span
spannable = new SpannableString(str);
}
for (int i = ; i < str.length(); i++) {
int index1 = str.indexOf("[", i);
int length1 = str.indexOf("]", index1 + );
int index2 = str.indexOf(":", i);
try {
String emojiStr = str.substring(index1, length1 + "]".length());
int resId = getEmojiResId(emojiStr);
if (resId > ) {
// 构建图片span
Drawable drawable = res.getDrawable(resId);
if (drawable != null) {
int itemHeight;
int itemWidth;
if (fontSize == WRAP_DRAWABLE) {
itemHeight = drawable.getIntrinsicHeight();
itemWidth = drawable.getIntrinsicWidth();
} else {
itemHeight = fontSize;
itemWidth = fontSize;
}
drawable.setBounds(, , itemHeight, itemWidth);
// 自定义ImageSpan,实现文字和表情的对齐
EmojiSpan imageSpan = new EmojiSpan(drawable,lineSpace);
spannable.setSpan(imageSpan, index1, length1 + "]".length(), Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
}
}
} catch (Exception e) {
}
}
return spannable;
}

3.处理表情图片的工具类,主要是计算表情的显示大小和位置

public class EmojiSpan extends ImageSpan {
private int lineSpace;
public EmojiSpan(Drawable drawable,int lineSpace) {
super(drawable);
this.lineSpace = lineSpace;
}
public EmojiSpan(Context context, int resourceId) {
super(context, resourceId);
}
@Override
public int getSize(Paint paint, CharSequence text, int start, int end, Paint.FontMetricsInt fontMetricsInt) {
Drawable drawable = this.getDrawable();
Rect rect = drawable.getBounds();
if(fontMetricsInt != null) {
Paint.FontMetricsInt fmPaint = paint.getFontMetricsInt();
int fontHeight = fmPaint.bottom - fmPaint.top;
int drHeight = rect.bottom - rect.top;
int top = drHeight / - fontHeight / ;
int bottom = drHeight / + fontHeight / ;
fontMetricsInt.ascent = -bottom;
fontMetricsInt.top = -bottom;
fontMetricsInt.bottom = top;
fontMetricsInt.descent = top;
}
return rect.right;
}
@Override
public void draw(Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, Paint paint) {
Drawable drawable = this.getDrawable();
canvas.save();
int transY = (bottom - top - drawable.getBounds().bottom) / + top - lineSpace;
canvas.translate(x, (float)transY);
drawable.draw(canvas);
canvas.restore();
}
}

4.获取TextView字体大小的工具

/**
* 获取TextView的字体大小
* @param textView
* @return
*/
public static int getFontHeight(TextView textView) {
Paint paint = new Paint();
paint.setTextSize(textView.getTextSize());
Paint.FontMetrics fm = paint.getFontMetrics();
return (int) Math.ceil(fm.bottom - fm.top);
}

TextView展示富文本时emoj或图片和文字不对齐的解决方案的更多相关文章

  1. wordpress 后台富文本编辑器,添加图片发现无法左对齐,样式出现混乱

    如上图所示,无法左对齐,但是左对齐的按钮全部是正确的,最后一点点排除,发现是因为这个词的影响,去掉就好了,原因不明,可能是这个词被当做某个方法执行了

  2. 【css对齐】块内或者行内图片与文字居中对齐最靠谱的方式!

    块内或者行内图片与文字居中对齐最靠谱的方式! 做图片与文字在一行的按钮时候最常用到,总结了一个靠谱的方法,终于可以完美的对齐下面给个代码 首先是html: <p class="btnU ...

  3. TextView显示HTML文本时<IMG>标签指定图片的显示处理

    TextView显示文本时是支持一些HTML标签的(具体支持那些标签会在下面附录列出),不会需要先用HTML的static方法fromHtml来转换一下. Spanned text = Html.fr ...

  4. Simditor 富文本编辑器多选图片上传、视频连接插入

    simditor 是一个基于浏览器的所见即所得的文本编辑器.Simditor 富文本编辑器, 支持多选图片上传, 视频连接插入, HTML代码编辑以及常用富文本按钮,支持的浏览器:IE10.Firef ...

  5. TextView之富文本

    项目中使用富文本比较常见了,一行显示多种样式颜色的文本,使用 ClickableSpan 富文本实现在同一个 TextView 中的文本的颜色.大小.背景色等属性的多样化和个性化. 我们也可以使用Ht ...

  6. 富文本编辑器TInyMCE,本地图片上传(Image Upload)

    TinyMCE 官网 (类似:百度的富文本web编辑器UEditor) 第一步 下载 TinyMCE,解压后放入工程,在需要的HTML页面引入tinymce.min.js. 第二步 下载tinyMCE ...

  7. angularjs中展示富文本编辑器文本,向DOM中插入元素

    前几天在用textangular富文本编辑器插件时,将存储的文本及格式存储到数据库中,但是从后台接口中再向angular页面插入时却不能执行,即在Angular中操作DOM没有实现,后来查看了一下,操 ...

  8. wangEditor富文本编辑器使用及图片上传

    引入js文件 <script type="text/javascript" src="style/js/wangEditor.min.js">< ...

  9. iView + vue-quill-editor 实现一个富文本编辑器(包含图片,视频上传)

    1. 引入插件(注意IE10以下不支持) npm install vue-quill-editor --savenpm install quill --save (Vue-Quill-Editor需要 ...

随机推荐

  1. Python __slots__限制动态添加变量

    Python是一种非常灵活的动态语言,有时感觉太灵活以至于不知道遵循什么样的规则去驾驭.不过Python已经是非常完备的语言,想实现什么样的功能都是有方法的,而且也很容易,比如限制一个类动态添加成员变 ...

  2. vue.js简单添加和删除

    这只是个简单的添加和删除,没有连接后台数据的 <%@ page language="java" contentType="text/html; charset=UT ...

  3. JAVA程序调试

    调试 步骤1:设置断点(不能在空白处设置断点) 步骤2:启动调试 步骤3:调试代码(F6单步跳过)笔记本Fn+F6(F5) 步骤4:结束调试 掌握调试的好处? l  很清晰的看到,代码执行的顺序 l  ...

  4. Node.js 初识1

    测试:让Node.js运行脚本 1.创建一个脚本 js1.js console.log('测试'); 2.cmd界面 运行脚本

  5. psdTohtml

    https://github.com/anjorweb/fastHtml fastHtml 一个简单的psd直接导出html的工具 自己工作常用整理 适合单页面且采用DOM结构布局的H5页面,基于Ca ...

  6. Android 查阅博客1_app优化_1大小

    Android  App  Bundle (google play 商店发布应用的话,可自行深入了解下,这里不做介绍) http://mp.weixin.qq.com/s?__biz=MzAwODY4 ...

  7. DB2在dbvisualizer 客户端执行begi/end 语句块

    注意,begin end 代码块在dbvisualizer 执行前要加 --/   后面要加   / 注意,begin end 代码块在dbvisualizer 执行前要加 --/   后面要加    ...

  8. java基础 ---- 练习for循环

    -----   使用for循环打印图形 //打印矩形 public class Print { public static void main(String[] args) { for(int i=1 ...

  9. 650. 2 Keys Keyboard复制粘贴的次数

    [抄题]: Initially on a notepad only one character 'A' is present. You can perform two operations on th ...

  10. 激活prompt

    1.下载SQLPrompt 2. 断网, 打开注册机,拷贝验证码 2. 点击activate, 拷贝代码