Android自定义View之音频条形图

最近在学习Android自定义View,看到一个自定义音频条形图的例子,遂学习了一下并且在此基础上扩展了一点,在此记录一下,来帮助到需要的人。先放上一张效果图,看着还不错吧,接下来就开始一步步实现这个小例子。

音频条形图

先说说这个自定义View的基本思路,首先最主要其实就是绘制一个个小矩形,这里主要是涉及到绘制时的坐标计算,然后为了让其动起来,我们使用随机数来作为每个小矩形的高度,同时为了效果更逼真,我们还可以在每次View进行重绘时加一个延时,还可以利用LinearGradient来让不同的小矩形有一个颜色渐变,最后为了定制性更好,还可以增加自定义属性,大致思路就是这样,让我们一步步实现吧。

一、定义类继承自View并修改构造方法

我们必须创建一个类,来修改其中的构造方法之间的调用,让一个参数的构造方法中调用两个参数构造方法,在扫那个参数的构造方法中调用两个参数的构造方法,然后在第三个构造方法中写一些变量初始化的代码即可,这样,无论别人是通过代码,还是通过在xml文件中使用我们的自定义View还是使用了我们自定义View的自定义属性,这些代码都会被调用,要是大家还不太清楚三个构造方法在何时调用,可以看看下面的注释。

/**
 * 代码中直接new时调用
 * @param context
 */
public AudioBarGraph(Context context) {
    this(context, null);
}

/**
 * 在xml中使用自定义View并且没有自定义属性时调用
 * @param context
 * @param attrs
 */
public AudioBarGraph(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
}

/**
 * 在xml中使用自定义View并且使用自定义属性时调用
 * @param context
 * @param attrs
 * @param defStyleAttr
 */
public AudioBarGraph(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
}

二、在布局文件中引用自定义View

注意这里的引用必须使用全路径

<com.codekong.customview.view.AudioBarGraph
    android:id="@+id/id_abg"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"/>

三、定义成员变量以备后面使用

每个成员变量都有注释

//小矩形的数目
private int mRectCount;
//每个小矩形的宽度
private int mRectWidth;
//每个小矩形的高度
private int mRectHeight;
//每个小矩形之间的偏移量
private float mOffset;
//绘制小矩形的画笔
private Paint mPaint;
//View的宽度
private int mViewWidth;
//产生渐变效果
private LinearGradient mLinearGradient;
//每个小矩形的当前高度
private float[] mCurrentHeight;

//渐变色顶部颜色
private int mTopColor;
//渐变色底部颜色
private int mBottomColor;
//View重绘延时时间
private int mDelayTime;

四、在xml文件中定义自定义属性

attrs.xml文件中定义一些属性,一边后面使用时可以自由配置

<declare-styleable name="AudioBarGraph">
    <attr name="rectCount" format="integer"/>         <!-- 小矩形的数目 -->  
    <attr name="rectOffset" format="float"/>          <!-- 每一个小矩形之间的偏移量 -->  
    <attr name="topColor" format="color"/>            <!-- 一个小矩形渐变的顶部颜色 -->  
    <attr name="bottomColor" format="color"/>         <!-- 一个小矩形渐变的底部颜色 -->  
    <attr name="delayTime" format="integer"/>         <!-- 小矩形变化的延时时间(毫秒) -->  
</declare-styleable>

五、获取自定义属性,并初始化变量

我们在三个参数的构造方法中初始化画笔并且获取我们的自定义属性,由于TypedArray对象是共享的资源,所以在获取完值之后必须要调用recycle()方法来回收。

public AudioBarGraph(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    //初始化画笔设置抗锯齿
    mPaint = new Paint();
    mPaint.setAntiAlias(true);
    mPaint.setStyle(Paint.Style.FILL);

    TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.AudioBarGraph);
    mRectCount = ta.getInt(R.styleable.AudioBarGraph_rectCount, 10);
    mOffset = ta.getFloat(R.styleable.AudioBarGraph_rectOffset, 3);
    mDelayTime = ta.getInt(R.styleable.AudioBarGraph_delayTime, 300);
    mTopColor = ta.getColor(R.styleable.AudioBarGraph_topColor, Color.YELLOW);
    mBottomColor = ta.getColor(R.styleable.AudioBarGraph_bottomColor, Color.BLUE);
    ta.recycle();
}

六、重写onMeasure()方法

此处重写onMeasure()方法是为了让其在wrap_content的情况下有一个默认的宽高

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int widthMode = MeasureSpec.getMode(widthMeasureSpec);
    int heightMode = MeasureSpec.getMode(heightMeasureSpec);
    int widthSize = MeasureSpec.getSize(widthMeasureSpec);
    int heightSize = MeasureSpec.getSize(heightMeasureSpec);
    //默认的宽高
    int width = 200;
    int height = 200;

    setMeasuredDimension(widthMode == MeasureSpec.EXACTLY ? widthSize : width,
            heightMode == MeasureSpec.EXACTLY ? heightSize : height);
}

七、重写onSizeChanged()方法

onSizeChanged()方法中我们获取到View的宽高及初始化一个LinearGradient,为后面的绘制做准备

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    mViewWidth = getMeasuredWidth();
    mRectHeight = getMeasuredHeight();
    mRectWidth = (int) (mViewWidth * 0.6 / mRectCount);
    mLinearGradient = new LinearGradient(0, 0, mRectWidth, mRectHeight,
            mTopColor, mBottomColor, Shader.TileMode.CLAMP);
    //给画笔设置Shader
    mPaint.setShader(mLinearGradient);
}

八、重写onDraw()方法开始绘制View

前面准备了那么多,现在终于要开始绘制小矩形了,其实这里最主要的就是一个坐标的计算,对于每个小矩形的高度,我们后面会留有公开的方法可以让用户去设置,如果用户没有提供每个小矩形的高度,我们则将使用随机的高度。

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    if (mCurrentHeight != null){
        //使用者指定了每个小矩形当前的高度则使用
        for (int i = 0; i < mRectCount; i++) {
            int random = (int) (Math.random() * 50);
            canvas.drawRect((float) (mViewWidth * 0.4 / 2 + mRectWidth * i + mOffset), mCurrentHeight[i] + random,
                    (float) (mViewWidth * 0.4 / 2 + mRectWidth * (i + 1)), mRectHeight, mPaint);
        }
    }else{
        //没有指定则使用随机数的高度
        for (int i = 0; i < mRectCount; i++) {
            int currentHeight = 0;
            canvas.drawRect((float) (mViewWidth * 0.4 / 2 + mRectWidth * i + mOffset), currentHeight,
                    (float) (mViewWidth * 0.4 / 2 + mRectWidth * (i + 1)), mRectHeight, mPaint);
        }
    }
    //让View延时mDelayTime毫秒再重绘
    postInvalidateDelayed(mDelayTime);
}

九、公开方法使使用者可以定义每个小矩形的高度

/**
 * 公开方法设置小矩形高度
 * @param currentHeight
 */
public void setCurrentHeight(float[] currentHeight){
    mCurrentHeight = currentHeight;
}

十、使用自定义view

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <com.codekong.customview.view.AudioBarGraph
        android:id="@+id/id_abg"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:rectCount="20"/>
</LinearLayout>
final AudioBarGraph audioBarGraph = (AudioBarGraph) findViewById(R.id.id_abg);
final float[] m = new float[20];
new Thread(new Runnable() {
    @Override
    public void run() {
        while (true){
            for (int i = 0; i < m.length; i++) {
                m[i] = (float) (Math.random() * 100);
            }
            audioBarGraph.setCurrentHeight(m);
            try {
                Thread.sleep(300);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}).start();

上面为了模拟音频的高低,开了一个子线程用来产生随机数并且设置给我们的自定义View,真实的环境中我们可以获取正式的音频高度来进行设置

后记

这只是一个简单的自定义View,查看源代码及更多自定义View,大家可以查看我的GitHub地址, https://github.com/codekongs/CustomView 上面有详细的使用说明,欢迎大家start和fork,你也可以贡献自己的自定义View,使其更加丰富

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,179评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,229评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,032评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,533评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,531评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,539评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,916评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,813评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,568评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,654评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,354评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,937评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,918评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,152评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,852评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,378评论 2 342

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,386评论 25 707
  • 国内自定义View的文章汗牛充栋,但是,即便是你全部看完也未必掌握这一知识(实际上,我也看了很多,但是一旦涉及自定...
    SnowDragonYY阅读 1,631评论 3 36
  • 一、Android开发初体验 二、Android与MVC设计模式模型对象存储着应用的数据和业务逻辑。模型类通常用来...
    为梦想战斗阅读 874评论 0 3
  • 即使我有时忘了我最爱的是谁, 可是当醉酒之后经常嘟囔你的名字。 放弃你后,我曾啊,后悔过, 但让我再次选择的话,我...
    西琛公子_诗歌与我阅读 216评论 0 0
  • 枝头一点春意生 渡口风起雨来急 渔父怏怏 无意弄扁舟 起身待欲归 更吹落 多少前尘
    小灰狗的胖孙女和三明治阅读 211评论 0 0