Android:实现IM多人员组合的群组头像

本文介绍如何在Android中实现类似微信群组头像的组合效果,通过自定义ViewGroup并重写onMeasure方法计算宽高,利用Glide加载和显示子头像。最后,文章提及了如何进一步实现圆形头像的技巧。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

说明:

此头像类似微信群组头像,整个头像由组内前N位人员的头像组合而成,可用网络或本地图片进行组合,最终显示为一个头像整体,看效果图:



一、自定义整体头像的ViewGroup,计算并保存宽高(重写onMeasure):

@Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        mWidth = getWidth(widthMeasureSpec);
        mHeight = getHeight(heightMeasureSpec);
        setMeasuredDimension(mWidth, mHeight);
    }

    private int getWidth(int measureSpec) {
        int width = MIN_WIDTH_AND_HEIGHT;
        int specMode = MeasureSpec.getMode(measureSpec);
        int specSize = MeasureSpec.getSize(measureSpec);
        if (specMode == MeasureSpec.EXACTLY) {
            width = specSize;
        } else if (specMode == MeasureSpec.AT_MOST) {
            width = Math.min(MIN_WIDTH_AND_HEIGHT, specSize);
        }
        return width;
    }

    private int getHeight(int measureSpec) {
        int height = MIN_WIDTH_AND_HEIGHT;
        int specMode = MeasureSpec.getMode(measureSpec);
        int specSize = MeasureSpec.getSize(measureSpec);
        if (specMode == MeasureSpec.EXACTLY) {
            height = specSize;
        } else if (specMode == MeasureSpec.AT_MOST) {
            height = Math.min(MIN_WIDTH_AND_HEIGHT, specSize);
        }
        return height;
    }

二、布局子头像的View(重写onLayout,对每个子头像进行布局):

@Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        layoutChild();
    }

    private void layoutChild() {
        if (mImgUrls == null || mImgUrls.isEmpty()) {
            return;
        }
        for (int i = 0; i < mImgSize; i++) {
            ImageView itemV = (ImageView) getChildAt(i);
            int left = 0, top = 0, right = 0, bottom = 0;
            /*
            对每个item的View计算left、top、right、bottom四个值
             */
            ...
            itemV.layout(left, top, right, bottom);  //真正布局子头像位置
            showImage(itemV, mImgUrls.get(i));  //加载并显示子头像图片
        }
    }

三、加载并显示各子头像(使用Glide加载并显示每个子头像)

private void showImage(Context context, ImageView iv, String url) {
        if (TextUtils.isEmpty(url)) {
            Bitmap bmp = BitmapFactory.decodeResource(context.getResources(), R.mipmap.user_default_icon);
            iv.setImageBitmap(bmp);
            return;
        }
        Glide.with(context).load(url)
                .diskCacheStrategy(DiskCacheStrategy.ALL)
                .dontAnimate()
                .placeholder(R.mipmap.user_default_icon)
                .error(R.mipmap.user_default_icon)
                .into(iv);
    }


到此多图片组合头像已经完成,不过想要圈形的还需要进行以下步奏


四、裁剪整个群头像为圆形(重写dispatchDraw):

@Override
    protected void dispatchDraw(Canvas canvas) {
        Path path = new Path();
        path.addCircle(mWidth / 2, mHeight / 2, mWidth / 2, Path.Direction.CW);
        canvas.clipPath(path);
        canvas.drawColor(Color.TRANSPARENT);
        super.dispatchDraw(canvas);
        drawGroupView(canvas);
    }

    /**
     * 绘制各头像间隔线
     * @param canvas
     */
    private void drawGroupView(Canvas canvas) {
        /*
        计算各条线的x和y坐标值
         */
        float[] point1 = new float[2], point2 = new float[2];
        ...
        drawLine(canvas, point1, point2);
    }

    /**
     * 绘制直线
     */
    private void drawLine(Canvas canvas, float[] point1, float[] point2) {
        mPaint.reset();
        mPaint.setAntiAlias(true);
        mPaint.setStrokeWidth(mInterval);
        mPaint.setColor(Color.WHITE);
        mPaint.setStyle(Paint.Style.STROKE);
        canvas.drawLine(point1[0], point1[1], point2[0], point2[1], mPaint);
    }

五、暴露公共方法供外部调用:

/**
     * 设置图片url集合
     *
     * @param imgs 图片url集合
     */
    public void setImages(List<String> imgs) {
        if (imgs == null || imgs.isEmpty()) {
            return;
        }
        if (imgs.size() > MAX_SIZE) {
            imgs = imgs.subList(0, MAX_SIZE);
        }
        removeAllViews();
        mImgUrls = imgs;
        mImgSize = imgs.size();
        for (int i = 0; i < mImgSize; i++) {
            View v = getItemView(i);
            if (v == null) {
                return;
            }
            addView(v, generateDefaultLayoutParams());
        }
        requestLayout();
    }

    /**
     * 设置单个图片url
     *
     * @param img 图片url
     */
    public void setImageUrl(String img) {
        ArrayList imgUrls = new ArrayList<>();
        imgUrls.add(img);
        setImages(imgUrls);
    }

    /**
     * 生成一个头像布局
     */
    private ImageView getItemView(int position) {
        ...
    }

六、使用:

1.写一个布局文件放自定义群组头像控件:

<?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="match_parent"
              android:background="#f2f2f2"
              android:orientation="vertical">

    <com.yyh.im.ui.widget.HeadView
        android:id="@+id/cv_head"
        android:layout_width="150dp"
        android:layout_height="150dp"/>

</LinearLayout>

2.代码中群组头像控件显示图片:

@BindView(R2.id.cv_head)
    public HeadView mHeadCv;

    private String[] IMG_URL_LIST = {
            "https://ss3.bdstatic.com/70cFv8Sh_Q1YnxGkpoWK1HF6hhy/it/u=416954025,2289731303&fm=27&gp=0.jpg",
            "https://ss1.bdstatic.com/70cFuXSh_Q1YnxGkpoWK1HF6hhy/it/u=704997830,3098922597&fm=27&gp=0.jpg",
            "https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=1375449509,557259337&fm=27&gp=0.jpg",
            "https://ss2.bdstatic.com/70cFvnSh_Q1YnxGkpoWK1HF6hhy/it/u=2825392570,1862752263&fm=27&gp=0.jpg",
            "https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=3252489351,440833245&fm=27&gp=0.jpg",
            "https://ss0.bdstatic.com/70cFuHSh_Q1YnxGkpoWK1HF6hhy/it/u=3586311245,3082636880&fm=27&gp=0.jpg"
    };

    private void showImage(){
        ArrayList<String> list = new ArrayList<>();
        for (int i = 0; i < 6; i++) {
            list.add(IMG_URL_LIST[i]);
        }
        mHeadCv.setImageUrls(list);
    }




可生成圆形、方形、及方形的组合头像。项目地址:https://github.com/Pedroafa/avatar-android 效果图:如何使用:首先创建个ImageView<ImageView             android:id="@ id/roundedAvatar"             android:layout_height="fill_parent"             android:layout_width="fill_parent"/>2. //通过AvatarDrawableFactory生成各种形状的Drawable AvatarDrawableFactory avatarDrawableFactory = new AvatarDrawableFactory(getResources()); BitmapFactory.Options options = new BitmapFactory.Options(); options.inMutable = false; Bitmap avatar = BitmapFactory.decodeResource(getResources(), R.drawable.avatar, options); //圆形的 Drawable roundedAvatarDrawable = avatarDrawableFactory.getRoundedAvatarDrawable(avatar);//生成圆形的Drawable ImageView roundedAvatarView = (ImageView)rootView.findViewById(R.id.roundedAvatar); roundedAvatarView.setImageDrawable(roundedAvatarDrawable);其他形状的://圆形的且带边框的 Drawable borderedRoundedAvatarDrawable = avatarDrawableFactory.getBorderedRoundedAvatarDrawable(avatar); ImageView borderedRoundedAvatarView = (ImageView)rootView.findViewById(R.id.borderedRoundedAvatar); borderedRoundedAvatarView.setImageDrawable(borderedRoundedAvatarDrawable); //方形的 Drawable squaredAvatarDrawable = avatarDrawableFactory.getSquaredAvatarDrawable(avatar); ImageView squaredAvatarView = (ImageView)rootView.findViewById(R.id.squaredAvatar); squaredAvatarView.setImageDrawable(squaredAvatarDrawable); //俩个方形的组合头像 Drawable doubleSquaredAvatarDrawable = avatarDrawableFactory.getSquaredAvatarDrawable(avatar, avatar); ImageView doubleSquaredAvatarView = (ImageView)rootView.findViewById(R.id.doubleSquaredAvatar); doubleSquaredAvatarView.setImageDrawable(doubleSquaredAvatarDrawable); //三个方形的组合头像 Drawable tripleSquaredAvatarDrawable = avatarDrawableFactory.getSquaredAvatarDrawable(avatar, avatar, avatar); ImageView tripleSquaredAvatarView = (ImageView)rootView.findViewById(R.id.tripleSquaredAvatar); tripleSquaredAvatarView.setImageDrawable(tripleSquaredAvatarDrawable); //四个方形的组合头像 Drawable quadrupleSquaredAvatarDrawable = avatarDrawableFactory                     .getSquaredAvatarDrawable(avatar, avatar, avatar, avatar); ImageView quadrupleSquaredAvatarView = (ImageView)rootView.findViewById(R.id.quadrupleSquaredAvatar); quadrupleSquaredAvatarView.setImageDrawable(quadrupleSquaredAvatarDrawable);
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值