Android实现动态曲线绘制

我们在安卓开发中,有时会用到统计图表的功能,而曲线绘制是其中比较典型的一种,一般是利用给定的坐标点集和安卓自带的绘图模块进行绘制,直接得到的是一张完整的静态的曲线图。但有时,我们需要动态绘制一些曲线图,就像我们打开电脑的任务管理器,里面有一个CPU使用记录的动态变化的带网格的曲线图,对于这一类的曲线绘制,安卓SDK自带的绘图模块貌似就不那么好用了。

在这里,我就利用Handler+timer机制和第三方开发包achartengine实现动态绘制安卓手机充放电曲线的应用。

1、下载第三方开发包achartengine,也就是jar包,可以在网上找到下载源,我下载的是achartengine-1.1.0.jar;

2、在创建的应用工程文件中引入该jar包,以Eclipse为例方法是:在包资源管理器目录中右键单击已创建的项目名称——>构建路径——>配置构建路径——>Java构建路径——>库——>外部JAR,然后找到achartengine包的存放路径,找到achartengine-1.1.0.jar文件,点击打开即完成了该包的引用,这时会在项目目录下出现一个名为引用的库文件,里面就是我们刚才引入的jar包。

3、接下来就可以按正常步骤开发该应用了,需要注意的是:需要在主类中引入jar中绘图需要的库文件,如下:

import org.achartengine.ChartFactory;
   import org.achartengine.GraphicalView;
   import org.achartengine.chart.PointStyle;
   import org.achartengine.model.XYMultipleSeriesDataset;
   import org.achartengine.model.XYSeries;
   import org.achartengine.renderer.XYMultipleSeriesRenderer;
   import org.achartengine.renderer.XYSeriesRenderer;

同时需要在AndroidManifest.xml文件中加上语句:

<application
   android:icon="@drawable/ic_launcher"
   android:label="@string/app_name" >
<activity android:name="org.achartengine.GraphicalActivity" />

4、接下来需要定义一些绘图的变量和样式:

private XYSeries series;
              private XYMultipleSeriesDataset mDataset;
              private GraphicalView chart;
              private XYMultipleSeriesRenderer renderer;
              private Context context, context1;
              private int addX = -1, addY;
              int[] xv = new int[720];// X轴数组元素个数
              int[] yv = new int[720];// Y轴数组元素个数         
                  context = getApplicationContext();
              // 这里获得main界面上的布局,下面会把图表画在这个布局里面
              LinearLayout layout = (LinearLayout) findViewById(R.id.linearLayout1);
              // 这个类用来放置曲线上的所有点,是一个点的集合,根据这些点画出曲线
              series = new XYSeries(title);
              // 创建一个数据集的实例,这个数据集将被用来创建图表
              mDataset = new XYMultipleSeriesDataset();
              // 将点集添加到这个数据集中
              mDataset.addSeries(series);
              // 以下都是曲线的样式和属性等等的设置,renderer相当于一个用来给图表做渲染的句柄
              int color = Color.GREEN;
              PointStyle style = PointStyle.CIRCLE;
              renderer = buildRenderer(color, style, true);
 
              // 设置好图表的样式(横轴时间分钟值,纵轴电压毫伏值)
              setChartSettings(renderer, "X", "Y", 0, 240, 3400, 4400, Color.WHITE,Color.WHITE);
              // 生成图表
              chart = ChartFactory.getLineChartView(context, mDataset, renderer);
              // 将图表添加到布局中去
              layout.addView(chart, new LayoutParams(LayoutParams.FILL_PARENT, LayoutParams.WRAP_CONTENT));
 
               protected XYMultipleSeriesRenderer buildRenderer(int color,PointStyle style, boolean fill)// 配置绘图属性
              {
              XYMultipleSeriesRenderer renderer = new XYMultipleSeriesRenderer();
              // 设置图表中曲线本身的样式,包括颜色、点的大小以及线的粗细等
              XYSeriesRenderer r = new XYSeriesRenderer();
              r.setColor(color);
              r.setPointStyle(style);
              r.setFillPoints(fill);
              r.setLineWidth((float) 1);// 线粗尺寸
              renderer.addSeriesRenderer(r);
              return renderer;
}
 
protected void setChartSettings(XYMultipleSeriesRenderer renderer,
String xTitle, String yTitle, double xMin, double xMax,
double yMin, double yMax, int axesColor, int labelsColor) {
// 有关对图表的渲染可参看api文档
renderer.setChartTitle(title);
renderer.setXTitle(xTitle);
renderer.setYTitle(yTitle);
renderer.setXAxisMin(xMin);
renderer.setXAxisMax(xMax);
renderer.setYAxisMin(yMin);
renderer.setYAxisMax(yMax);
renderer.setAxesColor(axesColor);
renderer.setLabelsColor(labelsColor);
renderer.setShowGrid(true);
renderer.setGridColor(Color.GREEN);// 曲线颜色
renderer.setXLabels(20);
renderer.setYLabels(10);
renderer.setChartTitle("时间/电压变化曲线图");// 图表名称
renderer.setXTitle("时间(min)");// 横坐标名称
renderer.setYTitle("电压(mv)");// 纵坐标名称
renderer.setYLabelsAlign(Align.RIGHT);
renderer.setPointSize((float) 1.5);// 设置点的大小
renderer.setShowLegend(false);
renderer.setPanEnabled(true, false);// 设置滑动,这边是横向可以滑动,纵向不可滑动
renderer.setZoomEnabled(true, false);// 设置缩放,横向可以,纵向不可以
// renderer.setZoomLimits(new double[] { 0, 720, 3400, 4400 });//设置缩放的范围
// renderer.setPanLimits(new double[] { -0.5, 720, 3400, 4400
// });//设置拉动的范围
}

5、动态绘制的实现:动态绘制主要需要实现两方面的内容,一是更新绘图的方法、二是动态绘图时间任务的定义。

1)更新绘图方法:

private void updateChart()// 更新绘图方法</strong>
{
// 设置好下一个需要增加的节点
addX = (int) (addX + 1);
addY = BatteryV;
// 移除数据集中旧的点集
mDataset.removeSeries(series);
// 判断当前点集中到底有多少点,因为屏幕总共只能容纳240个,所以当点数超过240时,长度永远是240
int length = series.getItemCount();
if (length > 720) {
length = 720;
}
// 将旧的点集中x和y的数值取出来放入backup中,并且将x的值加1,造成曲线向右平移的效果
for (int i = 0; i < length; i++) {
xv[i] = (int) series.getX(i);
yv[i] = (int) series.getY(i);
}
// 点集先清空,为了做成新的点集而准备
series.clear();
// 将新产生的点首先加入到点集中,然后在循环体中将坐标变换后的一系列点都重新加入到点集中
// 这里可以试验一下把顺序颠倒过来是什么效果,即先运行循环体,再添加新产生的点
series.add(addX, addY);
for (int k = 0; k < length; k++) {
series.add(xv[k], yv[k]);
}
// 在数据集中添加新的点集
mDataset.addSeries(series);
// 视图更新,没有这一步,曲线不会呈现动态
// 如果在非UI主线程中,需要调用postInvalidate(),具体参考api
chart.invalidate();
}

2)时间任务定义:

handler = new Handler() {
@Override
public void handleMessage(Message msg) {
// 刷新图表
updateChart();// 电压变化曲线刷新
updateChart1();// 电量变化曲线刷新
super.handleMessage(msg);
}
};
task = new TimerTask()// 刷新绘图计时任务配置
{
@Override
public void run() {
Message message = new Message();
message.what = 1;
handler.sendMessage(message);
}
};
timer.schedule(task, 0, 60000);// 以一分钟为时间间隔运行 

再发一下完整的代码:

主类:

package com.cfzz.vcd;
 
import java.io.FileOutputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
import java.util.Timer;
import java.util.TimerTask;
import org.achartengine.ChartFactory;
import org.achartengine.GraphicalView;
import org.achartengine.chart.PointStyle;
import org.achartengine.model.XYMultipleSeriesDataset;
import org.achartengine.model.XYSeries;
import org.achartengine.renderer.XYMultipleSeriesRenderer;
import org.achartengine.renderer.XYSeriesRenderer;
 
//import com.cfzz.vqcd.R;
 
//import com.cfzz.wy.R;
 
import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.graphics.Bitmap;
import android.graphics.Color;
import android.graphics.Paint.Align;
import android.os.BatteryManager;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.os.PowerManager;
import android.view.KeyEvent;
import android.view.View;
import android.view.ViewGroup.LayoutParams;
import android.view.WindowManager;
import android.widget.Button;
import android.widget.LinearLayout;
import android.widget.TextView;
import android.widget.Toast;
import android.view.View.OnClickListener;
 
public class VoltageChangeDraw extends Activity {
 
    // protected static final Activity Activity = null;
    private PowerManager.WakeLock wl;
    private Timer timer = new Timer();
    private long mExitTime = 0;
    private TimerTask task, task1;
    private Handler handler;
    private String title = "Signal Strength";
    private XYSeries series, series1;
    private XYMultipleSeriesDataset mDataset, mDataset1;
    private GraphicalView chart, chart1;
    private XYMultipleSeriesRenderer renderer, renderer1;
    private Context context, context1;
    private int addX = -1, addY;
    private int addX1 = -1, addY1;
    private Button button;
    int[] xv = new int[720];// X轴数组元素个数
    int[] yv = new int[720];// Y轴数组元素个数
    int[] xv1 = new int[720];// X轴数组元素个数
    int[] yv1 = new int[720];// Y轴数组元素个数
    public TextView TV;
    // ScreenShot screenShot = new ScreenShot();
    private int BatteryN; // 目前电量
    private int BatteryV; // 电池电压
    // private double BatteryT; //电池温度
    private String BatteryStatus; // 电池状态
    private String BatteryTemp; // 电池使用情况
 
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        getWindow().setFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON,
                WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
        setContentView(R.layout.main);
        this.button = (Button) this.findViewById(R.id.my_button);// 截图按键定义
        // 截图按键监听
        this.button.setOnClickListener(new OnClickListener() {
            public void onClick(View v) {
                SimpleDateFormat sdf = new SimpleDateFormat(
                        "yyyy-MM-dd_HH-mm-ss", Locale.US);// 日期格式名定义
 
                String fname = "/sdcard/" + sdf.format(new Date()) + ".png";// 存储路径及文件名定义
 
                View view = v.getRootView();
 
                view.setDrawingCacheEnabled(true);
 
                view.buildDrawingCache();
 
                Bitmap bitmap = view.getDrawingCache();
 
                if (bitmap != null) {
                    System.out.println("bitmap got!");
 
                    try {
 
                        FileOutputStream out = new FileOutputStream(fname);
 
                        bitmap.compress(Bitmap.CompressFormat.PNG, 100, out);// 图片格式及质量输出
 
                        System.out.println("file" + fname + "output done.");
 
                    } catch (Exception e) {
 
                        e.printStackTrace();
 
                    }
 
                } else {
                    System.out.println("bitmap is NULL!");
                }
            }
        });
        // MyTag可以随便写,可以写应用名称等
        PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
        // 换成PowerManager.SCREEN_DIM_WAKE_LOCK会变暗)
        PowerManager.WakeLock wl = pm.newWakeLock(
                PowerManager.SCREEN_BRIGHT_WAKE_LOCK, "MyTest");
        wl.acquire();// 开启屏幕常亮
 
        TV = (TextView) findViewById(R.id.TV);// 电池信息打印控件定义
 
        // 注册一个系统 BroadcastReceiver,作为访问电池计量之用,这个不能直接在AndroidManifest.xml中注册
        registerReceiver(mBatInfoReceiver, new IntentFilter(
                Intent.ACTION_BATTERY_CHANGED));
 
        context = getApplicationContext();
        context1 = getApplicationContext();// 图2
        // 这里获得main界面上的布局,下面会把图表画在这个布局里面
        LinearLayout layout = (LinearLayout) findViewById(R.id.linearLayout1);
 
        // 这个类用来放置曲线上的所有点,是一个点的集合,根据这些点画出曲线
        series = new XYSeries(title);
        series1 = new XYSeries(title);// 图2
        // 创建一个数据集的实例,这个数据集将被用来创建图表
        mDataset = new XYMultipleSeriesDataset();
        mDataset1 = new XYMultipleSeriesDataset();
        // 将点集添加到这个数据集中
        mDataset.addSeries(series);
        mDataset1.addSeries(series1);// 图2
        // 以下都是曲线的样式和属性等等的设置,renderer相当于一个用来给图表做渲染的句柄
        int color = Color.GREEN;
        int color1 = Color.RED;
        PointStyle style = PointStyle.CIRCLE;
        renderer = buildRenderer(color, style, true);
        renderer1 = buildRenderer(color1, style, true);// 图2
 
        // 设置好图表的样式(横轴时间分钟值,纵轴电压毫伏值)
        setChartSettings(renderer, "X", "Y", 0, 240, 3400, 4400, Color.WHITE,
                Color.WHITE);
        setChart1Settings(renderer1, "X", "Y", 0, 240, 0, 100, Color.WHITE,
                Color.WHITE);
        // 生成图表
        chart = ChartFactory.getLineChartView(context, mDataset, renderer);
        chart1 = ChartFactory.getLineChartView(context1, mDataset1, renderer1);// 图2
        </span><span style="font-size:12px;">// 将图表添加到布局中去,此处的纵向尺寸设置存在一些问题,当两张图都设置<span style="font-family: Arial, Helvetica, sans-serif;">WRAP_CONTENT</span>时,在实际界面上只显示第一张图,在这里可根据手                  // </span><span style="font-family: Arial, Helvetica, sans-serif;">机屏幕的大小设置具</span><span style="font-family: Arial, Helvetica, sans-serif;">体的数值(像素点个数),可以保证两张图都能显示在屏幕上
</span><span style="font-size:14px;">        layout.addView(chart, new LayoutParams(LayoutParams.FILL_PARENT,LayoutParams.WRAP_CONTENT));
        layout.addView(chart1, new LayoutParams(LayoutParams.FILL_PARENT,LayoutParams.WRAP_CONTENT));
        // layout.addView(chart, new LayoutParams(LayoutParams.FILL_PARENT,
        // 400));//525
        // layout.addView(chart1, new LayoutParams(LayoutParams.FILL_PARENT,
        // 400));//图2 525
        // 这里的Handler实例将配合下面的Timer实例,完成定时更新图表的功能
        handler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                // 刷新图表
                updateChart();// 电压变化曲线刷新
                updateChart1();// 电量变化曲线刷新
                super.handleMessage(msg);
 
            }
        };
 
        task = new TimerTask()// 刷新绘图计时任务配置
        {
            @Override
            public void run() {
                Message message = new Message();
                message.what = 1;
                handler.sendMessage(message);
            }
        };
        timer.schedule(task, 0, 60000);// 以一分钟为时间间隔运行
 
        task1 = new TimerTask()// 保存绘图计时任务配置
        {
            @Override
            public void run() {
                if (BatteryStatus == "充满电"
                        || (BatteryStatus == "放电状态" && BatteryN == 1))// 充/放电完成条件设定
                {
                    button.callOnClick();// 保存曲线图按键触发
                    timer.cancel(); // 停止计时功能
                }
            }
        };
        timer.schedule(task1, 0, 10000);// 以十秒钟为时间间隔运行
    }
 
    @Override
    public void onDestroy() // 结束程序声明
    {
        button.callOnClick();// 保存曲线图按键触发
        // 当结束程序时关掉Timer
        timer.cancel();
        // wl.release();
        // wl = null;
        super.onDestroy();
    }
 
    protected XYMultipleSeriesRenderer buildRenderer(int color,
            PointStyle style, boolean fill)// 配置绘图属性
    {
        XYMultipleSeriesRenderer renderer = new XYMultipleSeriesRenderer();
 
        // 设置图表中曲线本身的样式,包括颜色、点的大小以及线的粗细等
        XYSeriesRenderer r = new XYSeriesRenderer();
        r.setColor(color);
        r.setPointStyle(style);
        r.setFillPoints(fill);
        r.setLineWidth((float) 1);// 线粗尺寸
        renderer.addSeriesRenderer(r);
 
        return renderer;
    }
 
    // 图2
    protected XYMultipleSeriesRenderer buildRenderer1(int color1,
            PointStyle style, boolean fill)// 配置绘图属性
    {
        XYMultipleSeriesRenderer renderer1 = new XYMultipleSeriesRenderer();// 图2
        // 设置图表中曲线本身的样式,包括颜色、点的大小以及线的粗细等
        XYSeriesRenderer r = new XYSeriesRenderer();
        r.setColor(color1);
        r.setPointStyle(style);
        r.setFillPoints(fill);
        r.setLineWidth((float) 1);// 线粗尺寸
        renderer1.addSeriesRenderer(r);
 
        return renderer1;
    }
 
    protected void setChartSettings(XYMultipleSeriesRenderer renderer,
            String xTitle, String yTitle, double xMin, double xMax,
            double yMin, double yMax, int axesColor, int labelsColor) {
        // 有关对图表的渲染可参看api文档
 
        renderer.setChartTitle(title);
        renderer.setXTitle(xTitle);
        renderer.setYTitle(yTitle);
        renderer.setXAxisMin(xMin);
        renderer.setXAxisMax(xMax);
        renderer.setYAxisMin(yMin);
        renderer.setYAxisMax(yMax);
        renderer.setAxesColor(axesColor);
        renderer.setLabelsColor(labelsColor);
        renderer.setShowGrid(true);
        renderer.setGridColor(Color.GREEN);// 曲线颜色
        renderer.setXLabels(20);
        renderer.setYLabels(10);
        renderer.setChartTitle("时间/电压变化曲线图");// 图表名称
        renderer.setXTitle("时间(min)");// 横坐标名称
        renderer.setYTitle("电压(mv)");// 纵坐标名称
        renderer.setYLabelsAlign(Align.RIGHT);
        renderer.setPointSize((float) 1.5);// 设置点的大小
        renderer.setShowLegend(false);
        renderer.setPanEnabled(true, false);// 设置滑动,这边是横向可以滑动,纵向不可滑动
        renderer.setZoomEnabled(true, false);// 设置缩放,横向可以,纵向不可以
        // renderer.setZoomLimits(new double[] { 0, 720, 3400, 4400 });//设置缩放的范围
        // renderer.setPanLimits(new double[] { -0.5, 720, 3400, 4400
        // });//设置拉动的范围
    }
 
    // 图2
    protected void setChart1Settings(XYMultipleSeriesRenderer renderer1,
            String xTitle, String yTitle, double xMin, double xMax,
            double yMin, double yMax, int axesColor, int labelsColor) {
        // 有关对图表的渲染可参看api文档
 
        renderer1.setChartTitle(title);
        renderer1.setXTitle(xTitle);
        renderer1.setYTitle(yTitle);
        renderer1.setXAxisMin(xMin);
        renderer1.setXAxisMax(xMax);
        renderer1.setYAxisMin(yMin);
        renderer1.setYAxisMax(yMax);
        renderer1.setAxesColor(axesColor);
        renderer1.setLabelsColor(labelsColor);
        renderer1.setShowGrid(true);
        renderer1.setGridColor(Color.GREEN);// 曲线颜色
        renderer1.setXLabels(20);
        renderer1.setYLabels(10);
        renderer1.setChartTitle("时间/电量变化曲线图");// 图表名称
        renderer1.setXTitle("时间(min)");// 横坐标名称
        renderer1.setYTitle("电量(%)");// 纵坐标名称
        renderer1.setYLabelsAlign(Align.RIGHT);
        renderer1.setPointSize((float) 1.5);// 设置点的大小
        renderer1.setShowLegend(false);
        renderer1.setPanEnabled(true, false);// 设置滑动,这边是横向可以滑动,纵向不可滑动
        renderer1.setZoomEnabled(true, false);// 设置缩放,横向可以,纵向不可以
        // renderer1.setZoomLimits(new double[] { 0, 720, 0, 100 });//设置缩放的范围
        // renderer1.setPanLimits(new double[] { -0.5, 720, 0, 100 });//设置拉动的范围
    }
 
    private void updateChart()// 更新绘图方法
    {
 
        // 设置好下一个需要增加的节点
        addX = (int) (addX + 1);
        addY = BatteryV;
        // 移除数据集中旧的点集
        mDataset.removeSeries(series);
        // 判断当前点集中到底有多少点,因为屏幕总共只能容纳240个,所以当点数超过240时,长度永远是240
        int length = series.getItemCount();
        if (length > 720) {
            length = 720;
        }
        // 将旧的点集中x和y的数值取出来放入backup中,并且将x的值加1,造成曲线向右平移的效果
        for (int i = 0; i < length; i++) {
            xv[i] = (int) series.getX(i);
            yv[i] = (int) series.getY(i);
        }
        // 点集先清空,为了做成新的点集而准备
        series.clear();
 
        // 将新产生的点首先加入到点集中,然后在循环体中将坐标变换后的一系列点都重新加入到点集中
        // 这里可以试验一下把顺序颠倒过来是什么效果,即先运行循环体,再添加新产生的点
        series.add(addX, addY);
        for (int k = 0; k < length; k++) {
            series.add(xv[k], yv[k]);
        }
        // 在数据集中添加新的点集
        mDataset.addSeries(series);
 
        // 视图更新,没有这一步,曲线不会呈现动态
        // 如果在非UI主线程中,需要调用postInvalidate(),具体参考api
        chart.invalidate();
    }
 
    // 图2
    private void updateChart1()// 更新绘图方法
    {
 
        // 设置好下一个需要增加的节点
        addX1 = (int) (addX1 + 1);
        addY1 = BatteryN;
        // 移除数据集中旧的点集
        mDataset1.removeSeries(series1);
        // 判断当前点集中到底有多少点,因为屏幕总共只能容纳240个,所以当点数超过240时,长度永远是240
        int length1 = series1.getItemCount();
        if (length1 > 720) {
            length1 = 720;
        }
        // 将旧的点集中x和y的数值取出来放入backup中,并且将x的值加1,造成曲线向右平移的效果
        for (int j = 0; j < length1; j++) {
            xv1[j] = (int) series1.getX(j);
            yv1[j] = (int) series1.getY(j);
        }
        // 点集先清空,为了做成新的点集而准备
        series1.clear();
 
        // 将新产生的点首先加入到点集中,然后在循环体中将坐标变换后的一系列点都重新加入到点集中
        // 这里可以试验一下把顺序颠倒过来是什么效果,即先运行循环体,再添加新产生的点
        series1.add(addX1, addY1);
        for (int l = 0; l < length1; l++) {
            series1.add(xv1[l], yv1[l]);
        }
        // 在数据集中添加新的点集
        mDataset1.addSeries(series1);
 
        // 视图更新,没有这一步,曲线不会呈现动态
        // 如果在非UI主线程中,需要调用postInvalidate(),具体参考api
        chart1.invalidate();
    }
 
    /* 创建电池状态广播接收器 */
    public BroadcastReceiver mBatInfoReceiver = new BroadcastReceiver() {
        public void onReceive(Context context, Intent intent) {
            String action = intent.getAction();
 
            // 如果捕捉到的action是ACTION_BATTERY_CHANGED, 就运行onBatteryInfoReceiver()
 
            if (Intent.ACTION_BATTERY_CHANGED.equals(action)) {
                BatteryN = intent.getIntExtra("level", 0); // 目前电量
                BatteryV = intent.getIntExtra("voltage", 0); // 电池电压
                // BatteryT = intent.getIntExtra("temperature", 0); //电池温度
 
                switch (intent.getIntExtra("status",
                        BatteryManager.BATTERY_STATUS_UNKNOWN)) {
                case BatteryManager.BATTERY_STATUS_CHARGING:
                    BatteryStatus = "充电状态";
                    break;
                case BatteryManager.BATTERY_STATUS_DISCHARGING:
                    BatteryStatus = "放电状态";
                    break;
                case BatteryManager.BATTERY_STATUS_NOT_CHARGING:
                    BatteryStatus = "未充电";
                    break;
                case BatteryManager.BATTERY_STATUS_FULL:
                    BatteryStatus = "充满电";
 
                    break;
                case BatteryManager.BATTERY_STATUS_UNKNOWN:
                    BatteryStatus = "未知道状态";
                    break;
                }
 
                switch (intent.getIntExtra("health",
                        BatteryManager.BATTERY_HEALTH_UNKNOWN)) {
                case BatteryManager.BATTERY_HEALTH_UNKNOWN:
                    BatteryTemp = "未知错误";
                    break;
                case BatteryManager.BATTERY_HEALTH_GOOD:
                    BatteryTemp = "状态良好";
                    break;
                case BatteryManager.BATTERY_HEALTH_DEAD:
                    BatteryTemp = "电池没有电";
 
                    break;
                case BatteryManager.BATTERY_HEALTH_OVER_VOLTAGE:
                    BatteryTemp = "电池电压过高";
                    break;
                case BatteryManager.BATTERY_HEALTH_OVERHEAT:
                    BatteryTemp = "电池过热";
                    break;
                }
                TV.setText("目前电量为" + BatteryN + "% --- " + BatteryStatus + "\n"
                        + "电压为" + BatteryV + "mV ----- " + BatteryTemp);
 
            }
        }
 
    };
 
    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event)// 程序按返回键退出处理
    {
        switch (keyCode) {
        case KeyEvent.KEYCODE_BACK:
            // 双击退出
            // if (isStart == true){isStart = false;}//关闭前确认频闪是否关闭
            if ((System.currentTimeMillis() - mExitTime) > 2000) {
                Toast.makeText(this, "再按一次退出", Toast.LENGTH_SHORT).show();
                mExitTime = System.currentTimeMillis();
            } else {
                // isStart = false;
                finish();
            }
            return true;
 
        default:
            break;
        }
        return super.onKeyDown(keyCode, event);
    }
 
}

布局文件:

<?xml version="1.0" encoding="utf-8"?>  
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
    android:id="@+id/linearLayout1"
    
    android:orientation="vertical"  
    android:layout_width="fill_parent"  
    android:layout_height="fill_parent"
    android:layout_weight="1"
    >  
<!--  -->
<TextView  
    android:id = "@+id/TV" 
      
    android:layout_width="fill_parent"  
    android:layout_height="wrap_content" 
    
    />     
<Button
  android:text="保存该曲线图"
  android:textSize="4pt"
  android:id="@+id/my_button"
  
  android:layout_width="fill_parent"
  android:layout_height="28dp"
  android:layout_alignBottom="@+id/TV">
  <!--android:layout_alignParentBottom="true"-->
  
</Button>
</LinearLayout>

AndroidManifest.xml配置文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.cfzz.vcd"
    android:versionCode="1"
    android:versionName="1.0"> 
    
    <uses-sdk android:minSdkVersion="10" />
    <uses-permission android:name="android.permission.READ_FRAME_BUFFER"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.WAKE_LOCK" />
 
    <application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <activity android:name="org.achartengine.GraphicalActivity" />
        <activity
            android:name="com.cfzz.vcd.VoltageChangeDraw"
            android:screenOrientation="portrait"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
 
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
 
</manifest>

在该源码中,我们绘制了两条曲线,一条是电压变化曲线,另一条是电量变化曲线,并加一个保存手机屏幕的截图按钮和自动保存时间任务(也可手动保存),用来保存绘制的截图。

运行结果图(充电曲线图):

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

(0)

相关推荐

  • Android贝塞尔曲线初步学习第二课 仿QQ未读消息气泡拖拽黏连效果

    上一节初步了解了Android端的贝塞尔曲线,这一节就举个栗子练习一下,仿QQ未读消息气泡,是最经典的练习贝塞尔曲线的东东,效果如下 附上github源码地址:https://github.com/MonkeyMushroom/DragBubbleView 欢迎star~ 大体思路就是画两个圆,一个黏连小球固定在一个点上,一个气泡小球跟随手指的滑动改变坐标.随着两个圆间距越来越大,黏连小球半径越来越小.当间距小于一定值,松开手指气泡小球会恢复原来位置:当间距超过一定值之后,黏连小球消失,气泡小球

  • Android 利用三阶贝塞尔曲线绘制运动轨迹的示例

    本篇文章主要介绍了Android 利用三阶贝塞尔曲线绘制运动轨迹的示例,分享给大家,具体如下: 实现点赞效果,自定义起始点以及运动轨迹 效果图: xml布局: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/rl_root&

  • Android 曲线图的绘制示例代码

    本文介绍了Android 曲线图的绘制示例代码,分享给大家,具体如下: 效果展示 效果展示.gif 使用方式 // 初始化数据表格相关 with(mTableView) { // 配置坐标系 setupCoordinator("日", "人", /*这里是横坐标的值*/0f, 5f, 10f, 15f, 20f, 25f, 30f) // 添加曲线, 确保纵坐标的数值位数相等 addWave(ContextCompat.getColor(this@MainActiv

  • Android实现价格走势自定义曲线图

    本文是引用开源图表库框架 MPAndroidChart的LineChart 地址:https://github.com/PhilJay/MPAndroidChart 1.需求: (1)动态添加RadioButton,点击改变下面的LineChart数据 (2)LineChart绘制价格走势图,只显示最低点的小圆点和View,手指滑动,MarkView数据变化. (3) 服务端返回端数据,不是每一天端数据,但是x轴显示的必须是每一天的数据,这里是有我自己处理过的.返回里需要显示点的数组,之前的时间

  • Android把商品添加到购物车的动画效果(贝塞尔曲线)

    当我们写商城类的项目的时候,一般都会有加入购物车的功能,加入购物车的时候会有一些抛物线动画,具体代码如下: 实现效果如图: 思路: 确定动画的起终点 在起终点之间使用二次贝塞尔曲线填充起终点之间的点的轨迹 设置属性动画,ValueAnimator插值器,获取中间点的坐标 将执行动画的控件的x.y坐标设为上面得到的中间点坐标 开启属性动画 当动画结束时的操作 难点: PathMeasure的使用 - getLength() - boolean getPosTan(float distance, f

  • Android LineChart绘制多条曲线的方法

    本文实例为大家分享了Android LineChart绘制多条曲线的具体代码,供大家参考,具体内容如下 目标效果: 1.新建custom_marker_view.xml页面作为点击弹出框的页面: <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" andr

  • Android中贝塞尔曲线的绘制方法示例代码

    贝塞尔曲线,很多人可能不太了解,什么叫做贝塞尔曲线呢?这里先做一下简单介绍:贝塞尔曲线也可以叫做贝济埃曲线或者贝兹曲线,它由线段与节点组成,节点是可拖动的支点,线段像可伸缩的皮筋.一般的矢量图形软件常利用贝塞尔曲线来精确画出曲线. 上面的介绍中,"线段像可伸缩的皮筋"这句话非常关键,但也特别好理解.至于贝塞尔曲线的详细内容大家可以查阅相关资料.        Android提供的贝塞尔曲线绘制接口 在Android开发中,要实现贝塞尔曲线其实还是很简单的,因为Android已经给我们提

  • Android利用MPAndroidChart绘制曲线图表的基础教程

    前言 最近看了一下 Android 上的图表控件,去年做过一款应用也已上架了,也用到了图表控件,但是只是按照官方 demo 集成了,并没有过多的研究. 我最近基本上是面向 Github 编程了,拿别人造好的轮子拿来用,不能直接用的就改改再用,实在不好实现的就劝说产品去改需求,基本这是个套路. 既然是图表控件,就上 github 一番搜索,关键字 chart,按照 java 一筛选,再按照 start 数量排序,由高到低.基本上 start 从高到低的图表控件就是 MPAndroidChart.h

  • Android编程之canvas绘制各种图形(点,直线,弧,圆,椭圆,文字,矩形,多边形,曲线,圆角矩形)

    本文实例讲述了Android编程之canvas绘制各种图形的方法.分享给大家供大家参考,具体如下: 1.首先说一下canvas类: Class Overview The Canvas class holds the "draw" calls. To draw something, you need 4 basic components: A Bitmap to hold the pixels, a Canvas to host the draw calls (writing into

  • android贝塞尔曲线实现波浪效果

    本文实例为大家分享了android贝塞尔曲线实现波浪效果的具体代码,供大家参考,具体内容如下 因为手机录制gif不知道下什么软件好,所以暂时就先忽略效果图了 我在屏幕外多画了1.5个波浪,延伸至屏幕内.然后不断的循环,向右边移动.就有一种波浪的效果. 所以现在只需要画出左边的波长,然后再通过循环添加所有的波长即可. 第一个曲线已经确定了控制点和终点的坐标, 第二条曲线也可以很明显的看出来终点是在x轴的0点坐标,Y轴不变,而控制点是在负的波长的1/4的位置 有了上下曲线以后,其他的就可以直接通过循

随机推荐