100行Android代码轻松实现带动画柱状图

发布时间 - 2026-01-11 02:36:57    点击率:

为何要用带动画的柱状图呢?

最近,项目中遇到一个地方,要用到柱状图。所以这篇文章主要讲怎么搞一个柱子。100行代码,搞定柱状图!

圆角,头顶带数字。恩,这样用drawable也可以搞定。但是,这个柱子是有一个动画的,就是进入到界面的时候柱子不断的长高。这样的话,综合考虑还是用自定义View来做比较简便。效果如下图了:

完整Demo地址请到我的github下载地址:
https://github.com/lixiaodaoaaa/ColumnAnimViewProject

关于尺寸

控件尺寸直接来自xml中的设置,无需进行onMeasure测量。所以使用getWidth和getHeight获取高度。

关于数据范围

数据如果是一个柱子单独显示,则数据的范围不是很重要,但是柱状图通常是由很多柱子并列显示的,而这些柱子的单位高度都应该是一样的,所以提供设置最大值的范围,最小值就是0.

关于数字的文字大小

由于柱子的宽度就是整个View的宽度,所以数字的宽度不能超过柱子的宽度。因为这个原因,文字的size需要动态计算。意思就是 0和100000这两个数字显示的时候,文字的大小是不一样的。

关于边界值

0,是一个边界值(最小值),当显示0的时候,并不是柱子不显示的,而是显示一个最小高度的。

关于动画

不停的设置值,就会形成动画。意思是先设置数据1,然后紧接着设数据2.3.4.5……一直到最终的显示值,就会有动画效果。但是如果最终数值很大,1,1,1的增加就会很慢,动画时间很长。

完整代码如下:

package com.lixiaodaoaaa.view.pieview;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.RectF;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.view.View;

import com.gcssloop.graphics.R;
import com.lixiaodaoaaa.uitls.DensityUtils;

/**************************************
 * *** http://weibo.com/lixiaodaoaaa **
 * *** create at 2017/5/18  23:45 ****
 * ******* by:lixiaodaoaaa **********
 **************************************/

public class PColumn extends View {
  int MAX = 100;//最大
  int corner = 40;
  int data = 0;//显示的数
  int tempData = 0;
  int textPadding = 20;
  Paint mPaint;
  int mColor;

  Context mContext;

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

  public PColumn(Context context, @Nullable AttributeSet attrs) {
    super(context, attrs);
    mContext = context;
    initPaint();
  }

  public PColumn(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    mContext = context;
    initPaint();
  }

  private void initPaint() {
    mPaint = new Paint();
    mPaint.setAntiAlias(true);
    mColor = mContext.getResources().getColor(R.color.colorPrimary);
    mPaint.setColor(mColor);
  }

  @Override
  public void draw(Canvas canvas) {
    super.draw(canvas);

    if (data == 0) {
      mPaint.setTextSize(getWidth() / 2);
      RectF oval3 = new RectF(0, getHeight() - DensityUtils.pxTodip(mContext, 20), getWidth(), getHeight());// 设置个新的长方形
      canvas.drawRoundRect(oval3, DensityUtils.pxTodip(mContext, corner), DensityUtils.pxTodip(mContext, corner), mPaint);

      canvas.drawText("0",
          getWidth() * 0.5f - mPaint.measureText("0") * 0.5f,
          getHeight() - DensityUtils.pxTodip(mContext, 20) - 2 * DensityUtils.pxTodip(mContext, textPadding),
          mPaint);
      return;
    }

    //防止数值很大的的时候,动画时间过长
    int step = data / 100 + 1;

    if (tempData < data - step) {
      tempData = tempData + step;
    } else {
      tempData = data;
    }
    //画圆角矩形
    String S = tempData + "";
    //一个字和两,三个字的字号相同
    if (S.length() < 4) {
      mPaint.setTextSize(getWidth() / 2);
    } else {
      mPaint.setTextSize(getWidth() / (S.length() - 1));
    }

    float textH = mPaint.ascent() + mPaint.descent();
    float MaxH = getHeight() - textH - 2 * DensityUtils.pxTodip(mContext, textPadding);
    //圆角矩形的实际高度
    float realH = MaxH / MAX * tempData;
    RectF oval3 = new RectF(0, getHeight() - realH, getWidth(), getHeight());// 设置个新的长方形
    canvas.drawRoundRect(oval3, DensityUtils.pxTodip(mContext, corner), DensityUtils.pxTodip(mContext, corner), mPaint);
    //写数字
    canvas.drawText(S,
        getWidth() * 0.5f - mPaint.measureText(S) * 0.5f,
        getHeight() - realH - 2 * DensityUtils.pxTodip(mContext, textPadding),
        mPaint);
    if (tempData != data) {
      postInvalidate();
    }
  }

  public void setData(int data, int MAX) {
    this.data = data;
    tempData = 0;
    this.MAX = MAX;
    postInvalidate();
  }

}

/*
 * Copyright 2016 GcsSloop
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *  http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *
 * Last modified 2016-10-02 00:22:33
 *
 */

package com.lixiaodaoaaa.graphics;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;

import com.gcssloop.graphics.R;
import com.lixiaodaoaaa.view.pieview.PColumn;

public class MainActivity extends AppCompatActivity {

  private PColumn column_one;
  private PColumn column_two;
  private PColumn column_three;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    initAllViews();
  }

  private void initAllViews() {
    column_one = (PColumn) findViewById(R.id.column_one);
    column_two = (PColumn) findViewById(R.id.column_two);
    column_three = (PColumn) findViewById(R.id.column_three);

    column_one.setData(0, 100);
    column_two.setData(30, 100);
    column_three.setData(40, 100);

  }

}

xml配置如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:layout_weight="1"
  android:paddingBottom="@dimen/activity_vertical_margin"
  android:paddingLeft="@dimen/activity_horizontal_margin"
  android:paddingRight="@dimen/activity_horizontal_margin"
  android:paddingTop="@dimen/activity_vertical_margin"
  tools:context="com.lixiaodaoaaa.graphics.MainActivity"
>
  <View
    android:layout_width="0dp"
    android:layout_height="match_parent"
    android:layout_weight="0.2"/>
  <com.lixiaodaoaaa.view.pieview.PColumn
    android:id="@+id/column_one"
    android:layout_width="0dp"
    android:layout_height="match_parent"
    android:layout_weight="1"/>

  <View
    android:layout_width="0dp"
    android:layout_height="match_parent"
    android:layout_weight="2.4"/>

  <com.lixiaodaoaaa.view.pieview.PColumn
    android:id="@+id/column_two"
    android:layout_width="0dp"
    android:layout_height="match_parent"
    android:layout_weight="1"/>

  <View
    android:layout_width="0dp"
    android:layout_height="match_parent"
    android:layout_weight="2.4"/>

  <com.lixiaodaoaaa.view.pieview.PColumn
    android:id="@+id/column_three"
    android:layout_width="0dp"
    android:layout_height="match_parent"
    android:layout_weight="1"/>
  <View
    android:layout_width="0dp"
    android:layout_height="match_parent"
    android:layout_weight="0.2"/>
</LinearLayout>

完整Demo地址请到我的github下载地址:
https://github.com/lixiaodaoaaa/ColumnAnimViewProject

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# Android  # 柱状图  # Android自定义柱状图表的方法实例  # Android实现简易的柱状图和曲线图表实例代码  # Android自定义view实现动态柱状图  # 详解Android自定义View--自定义柱状图  # Android自定义带增长动画和点击弹窗提示效果的柱状图DEMO  # MPAndroidChart开源图表库的使用介绍之饼状图、折线图和柱状图  # Android自定义圆角柱状图  # 是一个  # 就会  # 下载地址  # 要用  # 圆角  # 请到  # 最小值  # 是有  # 是由  # 这两个  # 很重要  # 自定义  # 很长  # 一个字  # 这篇文章  # 来做  # 三个字  # 都应  # 不能超过 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何在云服务器上快速搭建个人网站?  Laravel如何使用Gate和Policy进行授权?(权限控制)  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  nginx修改上传文件大小限制的方法  JavaScript中的标签模板是什么_它如何扩展字符串功能  长沙企业网站制作哪家好,长沙水业集团官方网站?  微信公众帐号开发教程之图文消息全攻略  Python文件操作最佳实践_稳定性说明【指导】  长沙做网站要多少钱,长沙国安网络怎么样?  如何用美橙互联一键搭建多站合一网站?  猎豹浏览器开发者工具怎么打开 猎豹浏览器F12调试工具使用【前端必备】  西安专业网站制作公司有哪些,陕西省建行官方网站?  Laravel怎么返回JSON格式数据_Laravel API资源Response响应格式化【技巧】  如何在阿里云虚拟服务器快速搭建网站?  如何在阿里云完成域名注册与建站?  Win11怎样安装网易有道词典_Win11安装词典教程【步骤】  uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址  Laravel Blade模板引擎语法_Laravel Blade布局继承用法  Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】  Laravel路由怎么定义_Laravel核心路由系统完全入门指南  如何基于云服务器快速搭建网站及云盘系统?  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  网站制作软件有哪些,制图软件有哪些?  如何在阿里云服务器自主搭建网站?  java获取注册ip实例  详解jQuery中的事件  Laravel如何自定义分页视图?(Pagination示例)  Win11关机界面怎么改_Win11自定义关机画面设置【工具】  网站建设保证美观性,需要考虑的几点问题!  Laravel怎么实现观察者模式Observer_Laravel模型事件监听与解耦开发【指南】  Laravel如何与Inertia.js和Vue/React构建现代单页应用  邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?  音乐网站服务器如何优化API响应速度?  免费制作统计图的网站有哪些,如何看待现如今年轻人买房难的情况?  北京专业网站制作设计师招聘,北京白云观官方网站?  JS中对数组元素进行增删改移的方法总结  Laravel用户认证怎么做_Laravel Breeze脚手架快速实现登录注册功能  韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南  Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优  Laravel Asset编译怎么配置_Laravel Vite前端构建工具使用  微信小程序 HTTPS报错整理常见问题及解决方案  如何用JavaScript实现文本编辑器_光标和选区怎么处理  家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?  Laravel软删除怎么实现_Laravel Eloquent SoftDeletes功能使用教程  实例解析Array和String方法  如何快速生成专业多端适配建站电话?  Android GridView 滑动条设置一直显示状态(推荐)  b2c电商网站制作流程,b2c水平综合的电商平台?  Laravel如何配置任务调度?(Cron Job示例)  详解Huffman编码算法之Java实现