Отображение смайликов в Android

Мое приложение для обмена мгновенными сообщениями должно поддерживать смайлики. Они представляют собой GIF-файлы и имеют текстовые представления, которые используются в поле ввода, если пользователь выбирает одно из них. Но я хотел бы отображать их в виде изображений после того, как они были отправлены. В настоящее время мой пользовательский адаптер массива отображает отправленное сообщение в TextView строки.

Каков правильный метод динамического отображения изображений на основе их текстового представления? Должен ли я искать тексты смайликов, и если он найден, удалить TextView из макета (относительныйLayout подходит больше всего?) и добавить TextView с началом IM, ImageView со смайликом и еще один TextView. Если одновременно отправить больше смайликов, это может привести к беспорядку.

Есть ли более простой и логичный способ?


person Diepie    schedule 27.07.2010    source источник
comment
Я делаю что-то подобное ЗДЕСЬ !!! stackoverflow.com/questions/16768930 /   -  person toobsco42    schedule 27.05.2013


Ответы (2)



Я думаю, что было бы полезнее построить 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.

Изменить: обновленная версия значительно улучшила скорость.

person A-IV    schedule 29.11.2010
comment
Весь внутренний цикл for и удаление предыдущих интервалов не кажутся необходимыми, если вместо entry.getValue() вы используете entry.getValue().getConstantState().newDrawable() для получения клона чертежа для каждого своего внешнего вида. - person Gábor; 01.01.2015