免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2117 | 回复: 0
打印 上一主题 下一主题

[Android] android实现gif图与文字混排 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2015-06-11 09:28 |只看该作者 |倒序浏览
我们在进行qq聊天的时候发送表情,但这些表情都是并不是静态的,更多的是动态图,gif图,那么如何在android客户端显示动态gif图呢。

我们首先来看一下该开源项目的代码。该开源项目主要是通过自定义一个Adapter-------chatAdapter,在ChatAdapter每一条的setText属性中使用了自定义的方法convertNormalStringToSpannableString

convertNormalStringToSpannableString方法的返回值是SpannableString

  我们首先来了解一下什么是SpannableString

  TextView通常用来显示普通文本,但是有时候需要对其中某些文本进行样式、事件方面的设置。Android系统通过SpannableString类来对指定文本进行相关处理,也就是说我们想要实现文字加动态表情的实现就要通过SpannableString这个类来实现。
  1. private SpannableString convertNormalStringToSpannableString(String message , final TextView tv) {
  2.         SpannableString value = SpannableString.valueOf(message);
  3.         Matcher localMatcher = EMOTION_URL.matcher(value);
  4.         while (localMatcher.find()) {
  5.             String str2 = localMatcher.group(0);
  6.             int k = localMatcher.start();
  7.             int m = localMatcher.end();
  8.             if (m - k < 8) {
  9.                 int face = fm.getFaceId(str2);
  10.                 if(-1!=face){//wrapping with weakReference
  11.                     WeakReference<AnimatedImageSpan> localImageSpanRef = new WeakReference<AnimatedImageSpan>(new AnimatedImageSpan(new AnimatedGifDrawable(cxt.getResources().openRawResource(face), new AnimatedGifDrawable.UpdateListener() {   
  12.                         @Override
  13.                         public void update() {//update the textview
  14.                             tv.postInvalidate();
  15.                         }
  16.                     })));
  17.                     value.setSpan(localImageSpanRef.get(), k, m, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
  18.                 }
  19.             }
  20.         }
  21.         return value;
  22.     }
复制代码
首先将我们传入的message转化成SpannableString类,然后看一下传入的值是否符合我们一开始写好的正则表达式EMOTION_URL
  1. private Pattern EMOTION_URL = Pattern.compile("\\[(\\S+?)\\]");
复制代码
如果符合的话 我们取group(0)

  附:group是针对()来说的,group(0)就是指的整个串,group(1)指的是第一个括号里的东西,group(2)指的第二个括号里的东西。

  子表达式和起始位置和结束位置的差小于8,也就是符合我们的要求。调用FaceManager中的getFaceId方法
  1. public int getFaceId(String faceStr){
  2.         if(mFaceMap.containsKey(faceStr)){
  3.             return mFaceMap.get(faceStr);
  4.         }
  5.         return -1;
  6.     }
复制代码
找到我们用Map进行存储的表情

  如果表情存在的话利用一个弱引用(WeakReference)把自定义的AnimatedImageSpan进行处理,使AnimatedImageSpan不那么的消耗内存,在UpdateListener中利用postInvalidate刷新界面。最后把SpannableString的setSpan方法,三个参数分别是要放进去的span ,起始位置,结束位置,flag标志。

  关于flag:
  1. Spanned.SPAN_EXCLUSIVE_EXCLUSIVE, 这是在 setSpan 时需要指定的 flag,它是用来标识在 Span 范围内的文本前后输入新的字符时是否把它们也应用这个效果。分别有 Spanned.SPAN_EXCLUSIVE_EXCLUSIVE(前后都不包括)、 Spanned.SPAN_INCLUSIVE_EXCLUSIVE(前面包括,后面不包括)、 Spanned.SPAN_EXCLUSIVE_INCLUSIVE(前面不包括,后面包括)、 Spanned.SPAN_INCLUSIVE_INCLUSIVE(前后都包括)。
复制代码
最后将SpannableString返回,实现动态图文混排。

  关于自定义的AnimatedImageSpan如下:
  1. public class AnimatedImageSpan extends DynamicDrawableSpan {

  2.     private Drawable mDrawable;

  3.     public AnimatedImageSpan(Drawable d) {
  4.         super();
  5.         mDrawable = d;
  6.         // Use handler for 'ticks' to proceed to next frame
  7.         final Handler mHandler = new Handler();
  8.         mHandler.post(new Runnable() {
  9.             public void run() {
  10.                 ((AnimatedGifDrawable)mDrawable).nextFrame();
  11.                 // Set next with a delay depending on the duration for this frame
  12.                 mHandler.postDelayed(this, ((AnimatedGifDrawable)mDrawable).getFrameDuration());
  13.             }
  14.         });
  15.     }
  16.     @Override
  17.     public Drawable getDrawable() {
  18.         return ((AnimatedGifDrawable)mDrawable).getDrawable();
  19.     }

  20.     @Override
  21.     public int getSize(Paint paint, CharSequence text, int start, int end, Paint.FontMetricsInt fm) {
  22.         Drawable d = getDrawable();
  23.         Rect rect = d.getBounds();

  24.         if (fm != null) {
  25.             fm.ascent = -rect.bottom;
  26.             fm.descent = 0;

  27.             fm.top = fm.ascent;
  28.             fm.bottom = 0;
  29.         }

  30.         return rect.right;
  31.     }
  32.     @Override
  33.     public void draw(Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, Paint paint) {
  34.         Drawable b = getDrawable();
  35.         canvas.save();

  36.         int transY = bottom - b.getBounds().bottom;
  37.         if (mVerticalAlignment == ALIGN_BASELINE) {
  38.             transY -= paint.getFontMetricsInt().descent;
  39.         }

  40.         canvas.translate(x, transY);
  41.         b.draw(canvas);
  42.         canvas.restore();
  43.     }
  44. }
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP