37

我的 IM 应用程序必须支持表情符号。它们是 GIF 并具有文本表示,如果用户选择其中之一,则会在输入框中使用。但我想在发送它们后将它们显示为图像。目前我的自定义数组适配器在一行的 TextView 中显示发送的消息。

根据文本表示的出现动态显示图像的正确方法是什么?我是否必须搜索表情符号文本,如果找到,请从布局中删除 TextView(relativeLayout 最适合?)并添加一个带有 IM 开头的 TextView、一个带有表情符号的 ImageView 和另一个 TextView。如果同时发送更多表情符号,则可能会很混乱。

有没有更简单、更合乎逻辑的方法?

4

2 回答 2

122

我认为构建Spannable.

private static final Factory spannableFactory = Spannable.Factory
        .getInstance();

private static final Map<Pattern, Integer> emoticons = new HashMap<Pattern, Integer>();

static {
    addPattern(emoticons, ":)", R.drawable.emo_im_happy);
    addPattern(emoticons, ":-)", R.drawable.emo_im_happy);
    // ...
}

private static void addPattern(Map<Pattern, Integer> map, String smile,
        int resource) {
    map.put(Pattern.compile(Pattern.quote(smile)), resource);
}

public static boolean addSmiles(Context context, Spannable spannable) {
    boolean hasChanges = false;
    for (Entry<Pattern, Integer> entry : emoticons.entrySet()) {
        Matcher matcher = entry.getKey().matcher(spannable);
        while (matcher.find()) {
            boolean set = true;
            for (ImageSpan span : spannable.getSpans(matcher.start(),
                    matcher.end(), ImageSpan.class))
                if (spannable.getSpanStart(span) >= matcher.start()
                        && spannable.getSpanEnd(span) <= matcher.end())
                    spannable.removeSpan(span);
                else {
                    set = false;
                    break;
                }
            if (set) {
                hasChanges = true;
                spannable.setSpan(new ImageSpan(context, entry.getValue()),
                        matcher.start(), matcher.end(),
                        Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
            }
        }
    }
    return hasChanges;
}

public static Spannable getSmiledText(Context context, CharSequence text) {
    Spannable spannable = spannableFactory.newSpannable(text);
    addSmiles(context, spannable);
    return spannable;
}

实际上,此代码基于本机Html类的源代码。

编辑:更新版本具有显着的速度提升。

于 2010-11-29T09:04:39.053 回答
39

我会尝试使用正则表达式将每个表情符号的所有出现替换为<img>标签。然后,将该 HTML 转换为SpannedString viaHtml.fromHtml()SpannedString可以在通话setText()中使用TextView

于 2010-07-27T08:14:50.563 回答