基于Android MPAndroidChart实现腾讯QQ群数据统计报表核心功能
腾讯QQ移动客户端(新版)的QQ群有一项功能比较有趣,是关于QQ群的。QQ群新增一项功能开放给具有管理权限的群成员:管理群 -> 群数据 中,会看到QQ群的一些基础数据统计报表,如人数、发言条数的统计报表,如图:
我之前写了一篇文章是关于Android平台上的一个统计报表的开源框架MPAndroidChart,文章介绍了如何在自己的项目中使用MPAndroidChart制作统计报表,同时给出了基本折线图的一般用法(《Android统计图表MPAndroidChart》,链接地址:http://blog.csdn.net/zhangphil/article/details/47656521 )。我现在基于MPAndroidChart,在Android实现腾讯QQ群的统计报表的核心功能界面。现在给出实现的代码。
测试的MainActivity.java代码:
package zhangphil.chart;
import java.util.ArrayList;
import com.github.mikephil.charting.charts.LineChart;
import com.github.mikephil.charting.components.Legend;
import com.github.mikephil.charting.components.MarkerView;
import com.github.mikephil.charting.components.XAxis;
import com.github.mikephil.charting.components.XAxis.XAxisPosition;
import com.github.mikephil.charting.components.YAxis;
import com.github.mikephil.charting.data.Entry;
import com.github.mikephil.charting.data.LineData;
import com.github.mikephil.charting.data.LineDataSet;
import com.github.mikephil.charting.highlight.Highlight;
import com.github.mikephil.charting.utils.ValueFormatter;
import android.support.v7.app.ActionBarActivity;
import android.widget.TextView;
import android.content.Context;
import android.graphics.Color;
import android.os.Bundle;
/**
* @author Phil
*
*/
public class MainActivity extends ActionBarActivity {
private final int Y_BASE = 500;
private final String DATA_SET_LABEL = "发言条数。by ZhangPhil";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
LineChart chart = (LineChart) findViewById(R.id.chart);
// 弹出的数据点提示框。
MarkerView mv = new MyMarkerView(this, R.layout.marker_view);
chart.setMarkerView(mv);
// 制作7个数据点(沿x坐标轴)
LineData mLineData = makeLineData(7);
// 把X坐标轴放置到底部。默认的是在顶部。
XAxis xAxis = chart.getXAxis();
xAxis.setPosition(XAxisPosition.BOTTOM);
// X轴坐标线的颜色
xAxis.setAxisLineColor(Color.LTGRAY);
// false将不显示图表网格中的x轴标线
// xAxis.setEnabled(false);
// 不显示右边的Y坐标轴值
YAxis rightYAxis = chart.getAxisRight();
rightYAxis.setDrawLabels(false);
rightYAxis.setGridColor(Color.LTGRAY);
// 不显示左边的Y坐标轴值
YAxis leftYAxis = chart.getAxisLeft();
leftYAxis.setDrawLabels(false);
leftYAxis.setGridColor(Color.LTGRAY);
setChartStyle(chart, mLineData, Color.WHITE);
}
// 设置显示的样式
private void setChartStyle(LineChart mLineChart, LineData lineData,
int color) {
mLineChart.setDrawBorders(false);
mLineChart.setDescription(null);
// 如果没有数据的时候,会显示这个,类似listview的emtpyview
mLineChart
.setNoDataTextDescription("如果传给MPAndroidChart的数据为空,那么你将看到这段文字。@Zhang Phil");
// 是否绘制背景颜色。
// 如果mLineChart.setDrawGridBackground(false),
// 那么mLineChart.setGridBackgroundColor(Color.CYAN)将失效;
mLineChart.setDrawGridBackground(false);
mLineChart.setGridBackgroundColor(Color.CYAN);
// 触摸
mLineChart.setTouchEnabled(true);
// 拖拽
mLineChart.setDragEnabled(true);
// 缩放
mLineChart.setScaleEnabled(true);
mLineChart.setPinchZoom(false);
// 设置背景
mLineChart.setBackgroundColor(color);
// 设置x,y轴的数据
mLineChart.setData(lineData);
// 比例图标,y的value
Legend mLegend = mLineChart.getLegend();
// 如果设置了mLegend.setEnabled(false);
// 那么下面四行代码将失效。
// mLegend.setPosition(LegendPosition.BELOW_CHART_CENTER);
// mLegend.setForm(LegendForm.LINE);
// mLegend.setFormSize(8.0f);
// mLegend.setTextColor(Color.BLUE);
mLegend.setEnabled(false);
mLineChart.animateX(2000);
}
/**
* @param count
* 数据量,多少个数据。
* @return
*/
private LineData makeLineData(int count) {
ArrayList<String> x = new ArrayList<String>();
for (int i = 0; i < count; i++) {
// x轴显示的数据
x.add("8." + (i + 6));
}
// y轴的数据
ArrayList<Entry> y = new ArrayList<Entry>();
for (int i = 0; i < count; i++) {
int num = (int) (Math.random() * 500) + Y_BASE;
Entry entry = new Entry(num, i);
y.add(entry);
}
// y轴数据集
LineDataSet mLineDataSet = new LineDataSet(y, DATA_SET_LABEL);
// 用y轴的集合来设置参数
mLineDataSet.setLineWidth(4.0f);
mLineDataSet.setCircleSize(5.0f);
mLineDataSet.setColor(Color.GREEN);
mLineDataSet.setCircleColor(Color.GREEN);
// 设置mLineDataSet.setDrawHighlightIndicators(false)后,
// Highlight的十字交叉的纵横线将不会显示,
// 同时,mLineDataSet.setHighLightColor(Color.CYAN)失效。
mLineDataSet.setDrawHighlightIndicators(false);
mLineDataSet.setHighLightColor(Color.CYAN);
// 设置这项上显示的数据点的字体大小。
mLineDataSet.setValueTextSize(10.0f);
// mLineDataSet.setDrawCircleHole(true);
// 改变折线样式,用曲线。
mLineDataSet.setDrawCubic(true);
// 曲线的平滑度
mLineDataSet.setCubicIntensity(0.2f);
// 填充曲线下方的区域,红色,半透明。
mLineDataSet.setDrawFilled(true);
mLineDataSet.setFillAlpha(20);
mLineDataSet.setFillColor(Color.GREEN);
// 填充折线上数据点、圆球里面包裹的中心空白处的颜色。
mLineDataSet.setCircleColorHole(Color.WHITE);
// 设置折线上显示数据的格式。如果不设置,将默认显示float数据格式。
mLineDataSet.setValueFormatter(new ValueFormatter() {
@Override
public String getFormattedValue(float value) {
int n = (int) value;
String s = "" + n;
return s;
}
});
ArrayList<LineDataSet> mLineDataSets = new ArrayList<LineDataSet>();
mLineDataSets.add(mLineDataSet);
LineData mLineData = new LineData(x, mLineDataSets);
// 不要在折线上标出数据。
mLineData.setDrawValues(false);
return mLineData;
}
/**
* @author Phil
*
* 构造一个类似Android Toast的弹出消息提示框。
*/
private class MyMarkerView extends MarkerView {
private TextView tvContent;
public MyMarkerView(Context context, int layoutResource) {
super(context, layoutResource);
tvContent = (TextView) findViewById(R.id.tvContent);
}
@Override
public void refreshContent(Entry e, Highlight highlight) {
int n = (int) e.getVal();
tvContent.setText(n + "");
// if (e instanceof CandleEntry) {
// CandleEntry ce = (CandleEntry) e;
// tvContent.setText(""
// + Utils.formatNumber(ce.getHigh(), 0, true));
// } else {
// tvContent.setText("" + Utils.formatNumber(e.getVal(), 0, true));
// }
}
@Override
public int getXOffset() {
return -(getWidth() / 2);
}
@Override
public int getYOffset() {
return -getHeight();
}
}
}
MainActivity.java需要的activity_main.xml布局文件:
<RelativeLayout 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" >
<com.github.mikephil.charting.charts.LineChart
android:id="@+id/chart"
android:layout_width="match_parent"
android:layout_height="200dip" />
</RelativeLayout>
MainActivity.java中MyMarkerView类加载需要的布局文件marker_view.xml:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="50dip"
android:layout_height="30dip"
android:background="@drawable/marker" >
<TextView
android:id="@+id/tvContent"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_marginTop="5dp"
android:layout_marginLeft="5dp"
android:layout_marginRight="5dp"
android:textSize="10dp"
android:textColor="@android:color/white"
android:ellipsize="end"
android:singleLine="true" />
</RelativeLayout>
代码运行后,核心功能实现(细节的美工部分有待完善),结果如图: