Android显示自定义聊天表情

思路:

1、用RecycleView加载表情

2、每一个表情对应一个key

3、输入聊天框:自定义监听EditText获取到内容,找出内容包含的key,将key替换成图片

4、显示聊天记录:同理,自定义TextView

按照上面的思路完成功能:

第一步:引入图片

将所有的表情图片引入到drawable(或者加入压缩包,代码解压缩包,本地引入)

我是摸鱼摆烂,直接引入到drawable里的,哈哈哈哈

文件命名格式最好有规律,方便CV

第二步:创建表情List以及map

(List是RecycleView展示用,Map是监听文字内容包含的key,查找替换图片,用List还需循环)

创建实体类,方便引用

public class EmojiBean {
    private String code;
    private int icon;

    public EmojiBean(String code, int icon) {
        this.code = code;
        this.icon = icon;
    }

    public String getCode() {
        return code;
    }

    public void setCode(String code) {
        this.code = code;
    }

    public int getIcon() {
        return icon;
    }

    public void setIcon(int icon) {
        this.icon = icon;
    }

    @Override
    public String toString() {
        return "EmojiBean{" +
                "code='" + code + '\'' +
                ", icon=" + icon +
                '}';
    }
}

 创建List以及Map

import android.os.Build;

import com.okcis.linphone_demo.R;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

public class EmojiManager {
    public static final List<EmojiBean> EMOJI_LIST = new ArrayList<>();
    public static Map<String, Integer> EMOJI_MAP = new HashMap<>();


    public EmojiManager(){
        // 初始化表情数据
        EMOJI_LIST.add(new EmojiBean("[微笑]", R.drawable.emoji_0));
        EMOJI_LIST.add(new EmojiBean("[撇嘴]", R.drawable.emoji_1));
        //....省略
        EMOJI_LIST.add(new EmojiBean("[转圈]", R.drawable.emoji_108));

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            //将List转为Map,时代在发展,暂时不考虑N以下的设备
            EMOJI_MAP = EMOJI_LIST.stream().collect(Collectors.toMap
                    (EmojiBean::getCode, EmojiBean::getIcon, (value1, value2) -> value1));
        }
    }

    public static List<EmojiBean> getEmoticonList() {
        return EMOJI_LIST;
    }
    public static Map<String, Integer> getEmoticonMap() {
        return EMOJI_MAP;
    }
}

将整个功能模块封装 

加入到RecycleView

import android.content.Context;
import android.view.View;

import androidx.recyclerview.widget.GridLayoutManager;
import androidx.recyclerview.widget.RecyclerView;

import com.okcis.linphone_demo.R;
import com.okcis.linphone_demo.page.msg.emoji.data.EmojiBean;
import com.okcis.linphone_demo.page.msg.emoji.data.EmojiCallback;
import com.okcis.linphone_demo.page.msg.emoji.data.EmojiManager;
import com.okcis.linphone_demo.page.msg.emoji.data.EmojiAdapter;

import java.util.List;


public class EmojiView {
    private Context mContext;
    private View emojiView;
    private EmojiCallback mEmojiCallback;

    public EmojiView(Context context){
        mContext = context;
        initView();
    }

    private void initView(){
        emojiView = View.inflate(mContext, R.layout.chat_emoji_view_layout, null);
        EmojiManager emojiManager = new EmojiManager();
        RecyclerView recyclerView = emojiView.findViewById(R.id.recyclerView_emoji);
        List<EmojiBean> items = emojiManager.getEmoticonList();
        RecyclerView.LayoutManager layoutManager = new GridLayoutManager(mContext, 8);
        recyclerView.setLayoutManager(layoutManager);
        EmojiAdapter adapter = new EmojiAdapter(mContext, items);
        adapter.setOnItemClickListener((view, i) -> {
            if (mEmojiCallback != null){
                mEmojiCallback.EmojiResult(items.get(i));
            }
        });
        recyclerView.setAdapter(adapter);



    }

    public void setEmojiCallback(EmojiCallback callback){
        mEmojiCallback = callback;
    }

    public View getEmojiView(){
        return emojiView;
    }

}

chat_emoji_view_layout.xml 就一个recycleView

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">

    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/recyclerView_emoji"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:overScrollMode="never"
        android:padding="10dp"/>

</LinearLayout>

Adapter

import android.annotation.SuppressLint;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;

import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;

import com.bumptech.glide.Glide;
import com.luck.picture.lib.interfaces.OnItemClickListener;
import com.okcis.linphone_demo.R;

import java.util.List;

public class EmojiAdapter extends RecyclerView.Adapter<EmojiAdapter.ViewHolder> {
    private Context mContext;
    private OnItemClickListener mOnItemClickListener;
    private List<EmojiBean> mDataList;
    public EmojiAdapter(Context context, List<EmojiBean> mDataList) {
        super();
        mContext = context;
        this.mDataList = mDataList;
    }
    public void setOnItemClickListener(OnItemClickListener listener) {
        mOnItemClickListener = listener;
    }
    @NonNull
    @Override
    public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.chat_emoji_item_layout, parent, false);
        ViewHolder holder = new ViewHolder(view);
        return holder;
    }

    @Override
    public void onBindViewHolder(@NonNull ViewHolder holder, @SuppressLint("RecyclerView") int position) {
        EmojiBean data = mDataList.get(position);
        Glide.with(mContext).load(data.getIcon()).into(holder.DataIv);
        holder.DataIv.setOnClickListener(view -> {
            if (mOnItemClickListener != null){
                mOnItemClickListener.onItemClick(holder.DataIv, position);
            }
        });

    }

    @Override
    public int getItemCount() {
        if(mDataList != null){
            return mDataList.size();
        }
        return 0;
    }

    public class ViewHolder extends RecyclerView.ViewHolder {
        ImageView DataIv;

        public ViewHolder(@NonNull View itemView) {
            super(itemView);
            DataIv = itemView.findViewById(R.id.chat_emoji_icon);
        }
    }
}

chat_emoji_item_layout.xml,就一个图片

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center">

    <ImageView
        android:id="@+id/chat_emoji_icon"
        android:layout_width="28dp"
        android:layout_height="28dp"
        android:layout_marginVertical="8dp"/>

</LinearLayout>

点击表情Item,将key返回,EmojiCallback

public interface EmojiCallback {
    void EmojiResult(EmojiBean data);
}

 第三步:自定义EditText

import android.content.Context;
import android.graphics.drawable.Drawable;
import android.text.Editable;
import android.text.SpannableString;
import android.text.Spanned;
import android.text.TextWatcher;
import android.text.style.ImageSpan;
import android.util.AttributeSet;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.widget.AppCompatEditText;
import androidx.core.content.ContextCompat;

import com.okcis.linphone_demo.page.msg.emoji.data.EmojiManager;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

public class EmojiEditText extends AppCompatEditText {
    private Context mContext;
    private TextWatcher mTextWatcher;
    private Pattern pattern;
    private SpannableString spannableString;
    private Matcher matcher;

    public EmojiEditText(@NonNull Context context) {
        super(context);
        mContext = context;
        initView();
    }

    public EmojiEditText(@NonNull Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        mContext = context;
        initView();
    }

    public EmojiEditText(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        mContext = context;
        initView();
    }

    private void initView() {
        pattern = Pattern.compile("\\[(\\S+?)\\]");//正则,找出输入框符合key结构的字段
        mTextWatcher = new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence charSequence, int start, int count, int after) {
            }

            @Override
            public void onTextChanged(CharSequence charSequence, int start, int before, int count) {
                //移除监听,不移除下面有setText,会死循环
                removeTextChangedListener(mTextWatcher);
                String result = charSequence.toString();
                spannableString = new SpannableString(result);
                matcher = pattern.matcher(result);//找出内容中所有符合key规则的字段
                while (matcher.find()) {
                    if (EmojiManager.EMOJI_MAP.containsKey(matcher.group())) {
                        //Map根据key中找到图片
                        int icon = EmojiManager.EMOJI_MAP.get(matcher.group());
                        int startIndex = matcher.start();
                        int endIndex = matcher.end();
                        Drawable drawable = ContextCompat.getDrawable(mContext, icon);
                        //设置输入框内表情显示大小
                        drawable.setBounds(0, 0, (int) (drawable.getIntrinsicWidth() / 4), (int) (drawable.getIntrinsicHeight() / 4));
//                      有缺点,用ImageSpan表情图片和文字不会居中,并且会改变文字上下位置,因此需要自定义ImageSpan
//                     (自定义后也会有小浮动的改变,但还可以接受,不完美)
//                      ImageSpan imageSpan = new ImageSpan(drawable, ImageSpan.ALIGN_CENTER);
                        CenterAlignImageSpan imageSpan = new CenterAlignImageSpan(drawable, ImageSpan.ALIGN_CENTER);
                        spannableString.setSpan(imageSpan, startIndex, endIndex, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
                    }
                }
                setText(spannableString);
                setSelection(start + count);//光标移动,手动setText后光标会自动跳到最后面
                addTextChangedListener(mTextWatcher);//完成后再添加监听
            }

            @Override
            public void afterTextChanged(Editable editable) {
            }
        };
        addTextChangedListener(mTextWatcher);
    }

}

第四步:自定义TextView(与EditText同理,代码没有添加注释)

import android.content.Context;
import android.graphics.drawable.Drawable;
import android.text.Editable;
import android.text.SpannableString;
import android.text.Spanned;
import android.text.TextWatcher;
import android.text.style.ImageSpan;
import android.util.AttributeSet;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.widget.AppCompatTextView;
import androidx.core.content.ContextCompat;

import com.okcis.linphone_demo.page.msg.emoji.data.EmojiManager;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

public class EmojiTextView extends AppCompatTextView {
    private Context mContext;
    private TextWatcher mTextWatcher;
    private Pattern pattern;
    private SpannableString spannableString;
    private Matcher matcher;
    public EmojiTextView(@NonNull Context context) {
        super(context);
        mContext = context;
        initView();
    }

    public EmojiTextView(@NonNull Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        mContext = context;
        initView();
    }

    public EmojiTextView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        mContext = context;
        initView();
    }

    private void initView() {
        pattern = Pattern.compile("\\[(\\S+?)\\]");
        mTextWatcher = new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence charSequence, int start, int count, int after) {
            }

            @Override
            public void onTextChanged(CharSequence charSequence, int start, int before, int count) {
                removeTextChangedListener(mTextWatcher);
                String result = charSequence.toString();
                spannableString = new SpannableString(result);
                matcher = pattern.matcher(result);
                while (matcher.find()) {
                    if (EmojiManager.EMOJI_MAP.containsKey(matcher.group())) {
                        int startIndex = matcher.start();
                        int endIndex = matcher.end();
                        int icon = EmojiManager.EMOJI_MAP.get(matcher.group());
                        Drawable drawable = ContextCompat.getDrawable(mContext, icon);
                        drawable.setBounds(0, 0, (int) (drawable.getIntrinsicWidth() / 4), (int) (drawable.getIntrinsicHeight() / 4));
//                      有缺点,用ImageSpan表情图片和文字不会居中,并且会改变文字上下位置,因此需要自定义ImageSpan
//                     (自定义后也会有小浮动的改变,但还可以接受,不完美)
//                      ImageSpan imageSpan = new ImageSpan(drawable, ImageSpan.ALIGN_CENTER);
                        CenterAlignImageSpan imageSpan = new CenterAlignImageSpan(drawable, ImageSpan.ALIGN_CENTER);
                        spannableString.setSpan(imageSpan, startIndex, endIndex, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
                    }
                }
                setText(spannableString);
                addTextChangedListener(mTextWatcher);
            }

            @Override
            public void afterTextChanged(Editable editable) {
            }
        };
        addTextChangedListener(mTextWatcher);
    }
}

参考:CenterAlignImageSpan

import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.drawable.Drawable;
import android.text.style.ImageSpan;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;

public class CenterAlignImageSpan extends ImageSpan {

    /**
     * 图片和文字的左边距
     */
    private int mMarginLeft = 0;

    /**
     * 图片和文字的右边距
     */
    private int mMarginRight = 0;

    public CenterAlignImageSpan(@NonNull Drawable d) {
        super(d);
    }

    public CenterAlignImageSpan(@NonNull Drawable d, int verticalAlignment) {
        super(d, verticalAlignment);
    }

    public CenterAlignImageSpan(@NonNull Drawable d, int verticalAlignment, int marginLeft, int marginRight) {
        super(d, verticalAlignment);
        mMarginLeft = marginLeft;
        mMarginRight = marginRight;
    }

    @Override
    public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint) {
        Drawable drawable = getDrawable();
        canvas.save();
        Paint.FontMetricsInt fmPaint = paint.getFontMetricsInt();
        x = x + mMarginLeft;
        int fontHeight = fmPaint.descent - fmPaint.ascent;
        int centerY = y + fmPaint.descent - fontHeight / 2;
        int transY = centerY - (drawable.getBounds().bottom - drawable.getBounds().top) / 2;
        canvas.translate(x, transY);
        drawable.draw(canvas);
        canvas.restore();

    }

    @Override
    public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fontMetricsInt) {
        Drawable drawable = getDrawable();
        Rect rect = drawable.getBounds();
        if (fontMetricsInt != null) {
            Paint.FontMetricsInt fmPaint = paint.getFontMetricsInt();
            int fontHeight = fmPaint.descent - fmPaint.ascent;
            int drHeight = rect.bottom - rect.top;
            int centerY = fmPaint.ascent + fontHeight / 2;

            fontMetricsInt.ascent = centerY - drHeight / 2;
            fontMetricsInt.top = fontMetricsInt.ascent;
            fontMetricsInt.bottom = centerY + drHeight / 2;
            fontMetricsInt.descent = fontMetricsInt.bottom;
        }
        return mMarginLeft + rect.right + mMarginRight;
    }
}

 

 使用:

输入框使用EmojiEditText

聊天记录使用EmojiTextView

将EmojiView加入到预留的表情View位置(kotlin)

EmojiView(this).apply {
            setEmojiCallback {
                val sb = StringBuilder(binding.etContent.text.toString())
                val currentSelect = binding.etContent.selectionStart
                val msg = sb.insert(currentSelect, it.code)
                binding.etContent.setText(msg)
            }
        }.let {
            binding.emojiAddLayout.addView(it.emojiView)
        }

java

EmojiView emojiView = new EmojiView(this);
emojiView.setEmojiCallback(new EmojiCallback(){
        @Ov
        public void EmojiResult(EmojiBean data){
            //拼接输入框内容+表情key
            StringBuilder sb = new StringBuilder(EditText.getText().toString())//输入框内容
            int currentSelect = EditText.getSelectionStart()//当前光标所在位置
            String msg = sb.insert(currentSelect, data.code)//在光标位置插入表情key
            EditText.setText(msg)
        }
    }
)

 完活
效果图:

  • 9
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值