`
qq986945193
  • 浏览: 91086 次
  • 性别: Icon_minigender_2
  • 来自: 深圳
文章分类
社区版块
存档分类
最新评论

Android开发之制作圆形头像自定义View,直接引用工具类,加快开发速度。带有源代码学习

 
阅读更多

作者:程序员小冰,CSDN博客:http://blog.csdn.net/qq_21376985
QQ986945193 博客园主页:http://www.cnblogs.com/mcxiaobing/

Android开发之制作圆形头像自定义View,直接引用工具类,加快开发速度。带有源代码学习

大家都知道。现在好多头像都是圆形的,不再是以前的正方形或者长方形。
因此今天给大家带来的就是如何制作一个圆形头像。下面是效果图:
这里写图片描述
当然利用一个自定义View集成ImageView即可。

package com.qq986945193.zoomimageviewrounddemo;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Bitmap;
import android.graphics.Bitmap.Config;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.PorterDuff.Mode;
import android.graphics.PorterDuffXfermode;
import android.graphics.Rect;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.NinePatchDrawable;
import android.util.AttributeSet;
import android.widget.ImageView;

/**
 * 微博:http://weibo.com/mcxiaobing
 * ============================================================================
 * Copyright (c) 2015-2016 QQ986945193 All rights reserved.
 * ----------------------------------------------------------------------------
 * 类名:自定义圆形头像View
 * ----------------------------------------------------------------------------
 * 功能描述:http://blog.csdn.net/qq_21376985
 * ----------------------------------------------------------------------------
 */

public class RoundImageView extends ImageView {
    private int mBorderThickness = 0;
    private Context mContext;
    private int defaultColor = 0xFFFFFFFF;
    // 如果只有其中一个有值,则只画一个圆形边框
    private int mBorderOutsideColor = 0;// 图片的外边界
    private int mBorderInsideColor = 0;// 图片的内边界
    // 控件默认长、宽
    private int defaultWidth = 0;
    private int defaultHeight = 0;

    public RoundImageView(Context context) {
        super(context);
        mContext = context;
    }

    public RoundImageView(Context context, AttributeSet attrs) {
        super(context, attrs);
        mContext = context;
        setCustomAttributes(attrs);
    }

    public RoundImageView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        mContext = context;
        setCustomAttributes(attrs);
    }

    private void setCustomAttributes(AttributeSet attrs) {
        TypedArray a = mContext.obtainStyledAttributes(attrs,
                R.styleable.RoundImageView);
        // 边界的宽度
        mBorderThickness = a.getDimensionPixelSize(
                R.styleable.RoundImageView_border_thickness, 0);
        // 外边界的颜色
        mBorderOutsideColor = a.getColor(
                R.styleable.RoundImageView_border_outside_color, defaultColor);
        // 内边界的颜色
        mBorderInsideColor = a.getColor(
                R.styleable.RoundImageView_border_inside_color, defaultColor);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        Drawable drawable = getDrawable();
        if (drawable == null) {
            return;
        }

        if (getWidth() == 0 || getHeight() == 0) {
            return;
        }
        this.measure(0, 0);
        if (drawable.getClass() == NinePatchDrawable.class)
            return;
        Bitmap b = ((BitmapDrawable) drawable).getBitmap();
        Bitmap bitmap = b.copy(Config.ARGB_8888, true);
        if (defaultWidth == 0) {
            defaultWidth = getWidth();
        }
        if (defaultHeight == 0) {
            defaultHeight = getHeight();
        }
        // 保证重新读取图片后不会因为图片大小而改变控件宽、高的大小(针对宽、高为wrap_content布局的imageview,但会导致margin无效)
        // if (defaultWidth != 0 && defaultHeight != 0) {
        // LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(
        // defaultWidth, defaultHeight);
        // setLayoutParams(params);
        // }

        int radius = 0;
        if (mBorderInsideColor != defaultColor
                && mBorderOutsideColor != defaultColor) {// 定义画两个边框,分别为外圆边框和内圆边框
            radius = (defaultWidth < defaultHeight ? defaultWidth
                    : defaultHeight) / 2 - 2 * mBorderThickness;
            // 画内圆
            drawCircleBorder(canvas, radius + mBorderThickness / 2,
                    mBorderInsideColor);
            // 画外圆
            drawCircleBorder(canvas, radius + mBorderThickness
                    + mBorderThickness / 2, mBorderOutsideColor);
        } else if (mBorderInsideColor != defaultColor
                && mBorderOutsideColor == defaultColor) {// 定义画一个边框
            radius = (defaultWidth < defaultHeight ? defaultWidth
                    : defaultHeight) / 2 - mBorderThickness;
            drawCircleBorder(canvas, radius + mBorderThickness / 2,
                    mBorderInsideColor);
        } else if (mBorderInsideColor == defaultColor
                && mBorderOutsideColor != defaultColor) {// 定义画一个边框
            radius = (defaultWidth < defaultHeight ? defaultWidth
                    : defaultHeight) / 2 - mBorderThickness;
            drawCircleBorder(canvas, radius + mBorderThickness / 2,
                    mBorderOutsideColor);
        } else {// 没有边框
            radius = (defaultWidth < defaultHeight ? defaultWidth
                    : defaultHeight) / 2;
        }
        Bitmap roundBitmap = getCroppedRoundBitmap(bitmap, radius);
        canvas.drawBitmap(roundBitmap, defaultWidth / 2 - radius, defaultHeight
                / 2 - radius, null);
    }

    /**
     * 获取裁剪后的圆形图片
     * 
     * http://blog.csdn.net/qq_21376985
     * @param radius
     *            半径
     */
    public Bitmap getCroppedRoundBitmap(Bitmap bmp, int radius) {
        Bitmap scaledSrcBmp;
        int diameter = radius * 2;

        // 为了防止宽高不相等,造成圆形图片变形,因此截取长方形中处于中间位置最大的正方形图片
        int bmpWidth = bmp.getWidth();
        int bmpHeight = bmp.getHeight();
        int squareWidth = 0, squareHeight = 0;
        int x = 0, y = 0;
        Bitmap squareBitmap;
        if (bmpHeight > bmpWidth) {// 高大于宽
            squareWidth = squareHeight = bmpWidth;
            x = 0;
            y = (bmpHeight - bmpWidth) / 2;
            // 截取正方形图片
            squareBitmap = Bitmap.createBitmap(bmp, x, y, squareWidth,
                    squareHeight);
        } else if (bmpHeight < bmpWidth) {// 宽大于高
            squareWidth = squareHeight = bmpHeight;
            x = (bmpWidth - bmpHeight) / 2;
            y = 0;
            squareBitmap = Bitmap.createBitmap(bmp, x, y, squareWidth,
                    squareHeight);
        } else {
            squareBitmap = bmp;
        }

        if (squareBitmap.getWidth() != diameter
                || squareBitmap.getHeight() != diameter) {
            scaledSrcBmp = Bitmap.createScaledBitmap(squareBitmap, diameter,
                    diameter, true);

        } else {
            scaledSrcBmp = squareBitmap;
        }
        Bitmap output = Bitmap.createBitmap(scaledSrcBmp.getWidth(),
                scaledSrcBmp.getHeight(), Config.ARGB_8888);
        Canvas canvas = new Canvas(output);

        Paint paint = new Paint();
        Rect rect = new Rect(0, 0, scaledSrcBmp.getWidth(),
                scaledSrcBmp.getHeight());

        paint.setAntiAlias(true);
        paint.setFilterBitmap(true);
        paint.setDither(true);
        canvas.drawARGB(0, 0, 0, 0);
        canvas.drawCircle(scaledSrcBmp.getWidth() / 2,
                scaledSrcBmp.getHeight() / 2, scaledSrcBmp.getWidth() / 2,
                paint);
        paint.setXfermode(new PorterDuffXfermode(Mode.SRC_IN));
        canvas.drawBitmap(scaledSrcBmp, rect, rect, paint);
        // bitmap回收(recycle导致在布局文件XML看不到效果)
        // bmp.recycle();
        // squareBitmap.recycle();
        // scaledSrcBmp.recycle();
        bmp = null;
        squareBitmap = null;
        scaledSrcBmp = null;
        return output;
    }

    /**
     * 边缘画圆
     * 
     * http://blog.csdn.net/qq_21376985
     */
    private void drawCircleBorder(Canvas canvas, int radius, int color) {
        Paint paint = new Paint();
        /* 去锯齿 */
        paint.setAntiAlias(true);
        paint.setFilterBitmap(true);
        paint.setDither(true);
        paint.setColor(color);
        /* 设置paint的 style 为STROKE:空心 */
        paint.setStyle(Paint.Style.STROKE);
        /* 设置paint的外框宽度 */
        paint.setStrokeWidth(mBorderThickness);
        canvas.drawCircle(defaultWidth / 2, defaultHeight / 2, radius, paint);
    }

}

然后我们需要在布局中引用。下面是布局的一个引用,其实很简单,就和咱们
引用自定义View一模一样。由于它继承了ImageView。所以,方法大致相同。

 <com.qq986945193.zoomimageviewrounddemo.RoundImageView
        android:layout_width="100dp"
        android:layout_height="100dp"
        android:layout_centerInParent="true"
        android:scaleType="fitCenter"
        android:src="@drawable/img" />

当然还有,我们写了一个样式,所以需要在styles样式中添加:

   <!-- 圆形ImageView -->
    <declare-styleable name="RoundImageView">
        <attr name="border_thickness" format="dimension" />
        <attr name="border_inside_color" format="color" />
        <attr name="border_outside_color" format="color" />
    </declare-styleable>

OK,这样,圆形的的效果就能出来了,是不是很简单呢?当然,自定义View那一块想了解的可以深入。毕竟,你只会用,看不懂源代码也是不太好吧?是不是呢?
(Eclipse版本)源代码传送门:
http://download.csdn.net/detail/qq_21376985/9599249
(Android Studio版本)源代码传送门:
https://github.com/QQ986945193/DavidZoomImageViewRoundDemo

<script type="text/javascript"> $(function () { $('pre.prettyprint code').each(function () { var lines = $(this).text().split('\n').length; var $numbering = $('<ul/>').addClass('pre-numbering').hide(); $(this).addClass('has-numbering').parent().append($numbering); for (i = 1; i <= lines; i++) { $numbering.append($('<li/>').text(i)); }; $numbering.fadeIn(1700); }); }); </script>
分享到:
评论

相关推荐

    android开发中常用的工具类和自定义view

    android开发中常用的工具类和自定义viewandroid开发中常用的工具类和自定义viewandroid开发中常用的工具类和自定义viewandroid开发中常用的工具类和自定义viewandroid开发中常用的工具类和自定义viewandroid开发中...

    Android-Android开发之制作圆形头像图片的功能实现。自定义View实现圆形头像效果。

    在Android应用开发中,制作圆形头像图片是一个常见的需求,特别是在社交应用或者用户个人资料页面。这个功能的实现主要涉及到自定义View以及图片处理技术。本文将深入探讨如何通过自定义View来创建一个圆形的头像...

    android圆形头像自定义view

    "android圆形头像自定义view"就是一种实现方式,它允许我们通过自定义View来达到显示用户头像为圆形的效果,而不是传统的矩形。这种方法使得界面更加美观,尤其在社交应用、个人资料页等场景下常见。 在Android中,...

    android自定义圆形头像图片

    至此,我们就创建了一个自定义的圆形头像组件,它不仅能展示圆形图片,还能在底部显示带有自定义颜色、透明度和字体大小的文本。这个组件可以方便地集成到任何Android应用中,提升用户体验的同时,也展示了Android...

    Android开发-自定义View-AndroidStudio(一)

    在我们从Android小学生 进阶到 Android中学生的路上,肯定需要经历 自定义View。 大神已经给出了,最精简的Demo,我这里 顺水推舟,把一些基础知识 标记在代码之中,各位可以一边看源码 一边学基础。 觉得文章有用,...

    Android高手进阶之自定义View,自定义属性(带进度的圆形进度条)源码

    总结,自定义View与自定义属性是Android开发中的强大工具,它们能帮助开发者构建具有独特视觉效果和功能的组件。通过学习和实践如何创建一个带进度指示的圆形进度条,你可以提升自己的Android开发技能,并为你的应用...

    Android自定义View之绘制圆形头像功能

    在XML布局中声明自定义View时,系统会调用带有`AttributeSet`参数的构造方法,而在Java代码中动态创建实例时,根据传入的参数数量选择相应的构造方法。通过在各个构造方法中添加日志打印,可以验证这一点。 实现...

    Android显示圆形头像自定义类

    "Android显示圆形头像自定义类"这个主题就是关于如何在Android中创建一个能够将方形图片转换为圆形的自定义视图组件。下面将详细讲解这个知识点。 首先,要实现圆形头像的功能,我们需要创建一个新的自定义View类,...

    android 自定义view 圆形进度条

    在Android开发中,自定义View是一项重要的技能,它允许开发者根据特定需求创建独特且功能丰富的UI元素。本示例着重讲解如何实现一个圆形进度条,这个主题来自于xiaanming大神的源码改造,旨在帮助开发者理解并掌握...

    Android 自定义View实现环形带刻度的进度条

    在Android开发中,自定义View是一项重要的技能,它允许开发者根据特定需求创建独特且富有表现力的用户界面。本篇文章将深入探讨如何实现一个环形带刻度的进度条,这个自定义View适用于各种需要展示进度的情况,比如...

    Android 自定义View实现水平温度计

    - 自定义View通常通过继承已有的View或 ViewGroup 类,如在本项目中,可能会继承自`View`类。 - 需要重写`onDraw()`方法,这是绘制View的主要入口,所有图形绘制都在这里进行。 2. **图形绘制**: - 使用`Canvas...

    Android自定义View 圆形刻度罗盘 仪表盘 指针动态改变

    在Android开发中,自定义View是一项重要的技能,它允许开发者创造出独特且富有表现力的UI元素,以满足特定的设计需求。本项目聚焦于一个特定的自定义View:圆形刻度罗盘,也就是我们通常所说的仪表盘。这个仪表盘...

    Android自定义圆形SeekBar

    在Android开发中,SeekBar是一个非常常见的控件,用于让用户通过滑动来选择一个介于最小值和最大值之间的值。然而,系统默认的SeekBar是矩形的,有时为了追求更美观或者符合特定设计需求,开发者可能希望对其进行...

    Android 自定义各式各样的圆形ProgressBar

    要自定义圆形ProgressBar,我们需要创建一个新的View类,继承自Android的ProgressBar。这个自定义View将包含我们自己的绘图逻辑。以下是一些关键步骤: 1. **创建自定义View类**:新建一个Java类,例如`...

    安卓自定义控件相关-用自定义View的方式来实现圆形的遥控器菜单.rar

    在Android开发中,自定义控件是提升应用独特性和用户体验的重要手段。本示例主要讲解如何使用自定义View来实现一个圆形的遥控器菜单。这个圆形遥控器菜单可能用于模拟电视遥控器上的导航操作,或者作为游戏控制器等...

    自定义View系列源代码

    在Android开发中,自定义View是一项重要的技能,它允许开发者根据特定需求创建独特且功能丰富的UI元素。本项目"自定义View系列源代码"聚焦于几个特定的自定义视图,包括圆环、钟表、饼状图、雷达蜘蛛网以及流式布局...

    Android自定义View——圆形进度式按钮

    本文将详细解析"Android自定义View——圆形进度式按钮"这一主题,探讨如何构建一个具备进度显示功能且有特殊状态提示的自定义按钮。 首先,我们从标题"Android自定义View——圆形进度式按钮"可以理解,这个自定义...

    android自定义圆形布局CircleLayout

    在Android开发中,自定义布局是提升应用独特性和用户体验的重要手段。`CircleLayout`就是一种特殊的自定义布局,它使得内部的子视图按照圆形排列,增强了界面的视觉效果。本篇文章将深入探讨如何实现这样一个自定义...

    CircularImage自定义圆形头像,圆形图片

    总的来说,`CircularImage`自定义圆形头像是Android开发中的一个重要技巧,它涉及到图像处理、自定义View以及动画等多个方面的知识。掌握这个技巧,不仅可以提高应用的专业度,还能使我们在面对类似需求时更加...

Global site tag (gtag.js) - Google Analytics