Android自定义柱状图表的方法实例

前言

本文将通过示例代码介绍如何自定义简单的直方图表,此图表并非常见的直方图表,而是可以分组的。此文不会过多涉及原理,比较简单,示例图片如下(gif图片没有制作好,有闪烁,请见谅):

对于该示例的代码实现,其实重点在于坐标轴、文字、直方图的位置控制,需要随滑动距离而动态更新。注意事项会在示例代码中标注。下面贴出示例代码

public class MultiGroupHistogramView extends View {
 private int width;
 private int height;
 // 坐标轴线宽度
 private int coordinateAxisWidth;

 // 组名称字体大小
 private int groupNameTextSize;
 // 小组之间间距
 private int groupInterval;
 // 组内子直方图间距
 private int histogramInterval;
 private int histogramValueTextSize;
 // 图表数值小数点位数
 private int histogramValueDecimalCount;
 private int histogramHistogramWidth;
 private int chartPaddingTop;
 private int histogramPaddingStart;
 private int histogramPaddingEnd;
 // 各组名称到X轴的距离
 private int distanceFormGroupNameToAxis;
 // 直方图上方数值到直方图的距离
 private int distanceFromValueToHistogram;

 // 直方图最大高度
 private int maxHistogramHeight;
 // 轴线画笔
 private Paint coordinateAxisPaint;
 // 组名画笔
 private Paint groupNamePaint;
 private Paint.FontMetrics groupNameFontMetrics;
 private Paint.FontMetrics histogramValueFontMetrics;
 // 直方图数值画笔
 private Paint histogramValuePaint;
 // 直方图画笔
 private Paint histogramPaint;
 // 直方图绘制区域
 private Rect histogramPaintRect;
 // 直方图表视图总宽度
 private int histogramContentWidth;
 // 存储组内直方图shader color,例如,每组有3个直方图,该SparseArray就存储3个相对应的shader color
 private SparseArray<int[]> histogramShaderColorArray;

 private List<MultiGroupHistogramGroupData> dataList;
 private SparseArray<Float> childMaxValueArray;

 private Scroller scroller;
 private int minimumVelocity;
 private int maximumVelocity;
 private VelocityTracker velocityTracker;

 public MultiGroupHistogramView(Context context) {
  this(context, null);
 }

 public MultiGroupHistogramView(Context context, @Nullable AttributeSet attrs) {
  this(context, attrs, 0);
 }

 public MultiGroupHistogramView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
  super(context, attrs, defStyleAttr);
  init(attrs);
 }

 private void init(AttributeSet attrs) {
  setLayerType(View.LAYER_TYPE_HARDWARE, null);
  TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.MultiGroupHistogramView);
  coordinateAxisWidth = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_coordinateAxisWidth, DisplayUtil.dp2px(2));
  // 坐标轴线颜色
  int coordinateAxisColor = typedArray.getColor(R.styleable.MultiGroupHistogramView_coordinateAxisColor, Color.parseColor("#434343"));
  // 底部小组名称字体颜色
  int groupNameTextColor = typedArray.getColor(R.styleable.MultiGroupHistogramView_groupNameTextColor, Color.parseColor("#CC202332"));
  groupNameTextSize = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_groupNameTextSize, DisplayUtil.dp2px(15));
  groupInterval = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_groupInterval, DisplayUtil.dp2px(30));
  histogramInterval = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_histogramInterval, DisplayUtil.dp2px(10));
  // 直方图数值文本颜色
  int histogramValueTextColor = typedArray.getColor(R.styleable.MultiGroupHistogramView_histogramValueTextColor, Color.parseColor("#CC202332"));
  histogramValueTextSize = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_histogramValueTextSize, DisplayUtil.dp2px(12));
  histogramValueDecimalCount = typedArray.getInt(R.styleable.MultiGroupHistogramView_histogramValueDecimalCount, 0);
  histogramHistogramWidth = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_histogramHistogramWidth, DisplayUtil.dp2px(20));
  chartPaddingTop = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_chartPaddingTop, DisplayUtil.dp2px(10));
  histogramPaddingStart = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_histogramPaddingStart, DisplayUtil.dp2px(15));
  histogramPaddingEnd = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_histogramPaddingEnd, DisplayUtil.dp2px(15));
  distanceFormGroupNameToAxis = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_distanceFormGroupNameToAxis, DisplayUtil.dp2px(15));
  distanceFromValueToHistogram = typedArray.getDimensionPixelSize(R.styleable.MultiGroupHistogramView_distanceFromValueToHistogram, DisplayUtil.dp2px(10));
  typedArray.recycle();

  coordinateAxisPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
  coordinateAxisPaint.setStyle(Paint.Style.FILL);
  coordinateAxisPaint.setStrokeWidth(coordinateAxisWidth);
  coordinateAxisPaint.setColor(coordinateAxisColor);

  groupNamePaint = new Paint(Paint.ANTI_ALIAS_FLAG);
  groupNamePaint.setTextSize(groupNameTextSize);
  groupNamePaint.setColor(groupNameTextColor);
  groupNameFontMetrics = groupNamePaint.getFontMetrics();

  histogramValuePaint = new Paint(Paint.ANTI_ALIAS_FLAG);
  histogramValuePaint.setTextSize(histogramValueTextSize);
  histogramValuePaint.setColor(histogramValueTextColor);
  histogramValueFontMetrics = histogramValuePaint.getFontMetrics();

  histogramPaintRect = new Rect();
  histogramPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
  scroller = new Scroller(getContext(), new LinearInterpolator());
  ViewConfiguration configuration = ViewConfiguration.get(getContext());
  minimumVelocity = configuration.getScaledMinimumFlingVelocity();
  maximumVelocity = configuration.getScaledMaximumFlingVelocity();
 }

 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  width = getMeasuredWidth();
  height = getMeasuredHeight();
  maxHistogramHeight = height - groupNameTextSize - coordinateAxisWidth - distanceFormGroupNameToAxis - distanceFromValueToHistogram - histogramValueTextSize - chartPaddingTop;
 }

 /**
  * 判断是否可以水平滑动
  * @param direction 标识滑动方向 正数:右滑(手指从右至左移动);负数:左滑(手指由左向右移动)
  * 您可参考ScaollView或HorizontalScrollView理解滑动方向
  */
 @Override
 public boolean canScrollHorizontally(int direction) {
  if (direction > 0) {
   return histogramContentWidth - getScrollX() - width + histogramPaddingStart + histogramPaddingEnd > 0;
  } else {
   return getScrollX() > 0;
  }
 }

 /**
  * 根据滑动方向获取最大可滑动距离
  * @param direction 标识滑动方向 正数:右滑(手指从右至左移动);负数:左滑(手指由左向右移动)
  * 您可参考ScaollView或HorizontalScrollView理解滑动方向
  */
 private int getMaxCanScrollX(int direction) {
  if (direction > 0) {
   return histogramContentWidth - getScrollX() - width + histogramPaddingStart + histogramPaddingEnd > 0 ?
     histogramContentWidth - getScrollX() - width + histogramPaddingStart + histogramPaddingEnd : 0;
  } else if (direction < 0) {
   return getScrollX();
  }
  return 0;
 }

 private float lastX;

 @Override
 public boolean onTouchEvent(MotionEvent event) {
  initVelocityTrackerIfNotExists();
  velocityTracker.addMovement(event);
  switch (event.getAction()) {
   case MotionEvent.ACTION_DOWN: {
    if (!scroller.isFinished()) {
     scroller.abortAnimation();
    }
    lastX = event.getX();
    return true;
   }
   case MotionEvent.ACTION_MOVE: {
    int deltaX = (int) (event.getX() - lastX);
    lastX = event.getX();
    // 滑动处理
    if (deltaX > 0 && canScrollHorizontally(-1)) {
     scrollBy(-Math.min(getMaxCanScrollX(-1), deltaX), 0);
    } else if (deltaX < 0 && canScrollHorizontally(1)) {
     scrollBy(Math.min(getMaxCanScrollX(1), -deltaX), 0);
    }
    break;
   }
   case MotionEvent.ACTION_UP: {
    velocityTracker.computeCurrentVelocity(1000, maximumVelocity);
    int velocityX = (int) velocityTracker.getXVelocity();
    fling(velocityX);
    recycleVelocityTracker();
    break;
   }
   case MotionEvent.ACTION_CANCEL: {
    recycleVelocityTracker();
    break;
   }
  }
  return super.onTouchEvent(event);
 }

 private void initVelocityTrackerIfNotExists() {
  if (velocityTracker == null) {
   velocityTracker = VelocityTracker.obtain();
  }
 }

 private void recycleVelocityTracker() {
  if (velocityTracker != null) {
   velocityTracker.recycle();
   velocityTracker = null;
  }
 }

 // ACTION_UP事件触发
 private void fling(int velocityX) {
  if (Math.abs(velocityX) > minimumVelocity) {
   if (Math.abs(velocityX) > maximumVelocity) {
    velocityX = maximumVelocity * velocityX / Math.abs(velocityX);
   }
   scroller.fling(getScrollX(), getScrollY(), -velocityX, 0, 0, histogramContentWidth + histogramPaddingStart - width, 0, 0);
  }
 }

 @Override
 public void computeScroll() {
  if (scroller.computeScrollOffset()) {
   scrollTo(scroller.getCurrX(), 0);
  }
 }

 public void setDataList(@NonNull List<MultiGroupHistogramGroupData> dataList) {
  this.dataList = dataList;
  if (childMaxValueArray == null) {
   childMaxValueArray = new SparseArray<>();
  } else {
   childMaxValueArray.clear();
  }
  histogramContentWidth = 0;
  for (MultiGroupHistogramGroupData groupData : dataList) {
   List<MultiGroupHistogramChildData> childDataList = groupData.getChildDataList();
   if (childDataList != null && childDataList.size() > 0) {
    for (int i = 0; i < childDataList.size(); i++) {
     histogramContentWidth += histogramHistogramWidth + histogramInterval;
     MultiGroupHistogramChildData childData = childDataList.get(i);
     Float childMaxValue = childMaxValueArray.get(i);
     if (childMaxValue == null || childMaxValue < childData.getValue()) {
      childMaxValueArray.put(i, childData.getValue());
     }
    }
    histogramContentWidth += groupInterval - histogramInterval;
   }
  }
  histogramContentWidth += -groupInterval;
 }

 /**
  * 设置组内直方图颜色(并不是设置所有直方图颜色,而是根据每组数据内直方图数量设置)
  */
 public void setHistogramColor(int[]... colors) {
  if (colors != null && colors.length > 0) {
   if (histogramShaderColorArray == null) {
    histogramShaderColorArray = new SparseArray<>();
   } else {
    histogramShaderColorArray.clear();
   }
   for (int i = 0; i < colors.length; i++) {
    histogramShaderColorArray.put(i, colors[i]);
   }
  }
 }

 @Override
 protected void onDraw(Canvas canvas) {
  if (width == 0 || height == 0) {
   return;
  }
  int scrollX = getScrollX();
  int axisBottom = height - groupNameTextSize - distanceFormGroupNameToAxis - coordinateAxisWidth / 2;
  canvas.drawLine(coordinateAxisWidth / 2 + scrollX, 0, coordinateAxisWidth / 2 + scrollX, axisBottom, coordinateAxisPaint);
  canvas.drawLine(scrollX, axisBottom, width + scrollX, axisBottom, coordinateAxisPaint);
  if (dataList != null && dataList.size() > 0) {
   int xAxisOffset = histogramPaddingStart; // 每个直方图在x轴的偏移量
   for (MultiGroupHistogramGroupData groupData : dataList) {
    List<MultiGroupHistogramChildData> childDataList = groupData.getChildDataList();
    if (childDataList != null && childDataList.size() > 0) {
     int groupWidth = 0;
     for (int i = 0; i < childDataList.size(); i++) {
      MultiGroupHistogramChildData childData = childDataList.get(i);
      histogramPaintRect.left = xAxisOffset;
      histogramPaintRect.right = histogramPaintRect.left + histogramHistogramWidth;
      int childHistogramHeight;
      if (childData.getValue() <= 0 || childMaxValueArray.get(i) <= 0) {
       childHistogramHeight = 0;
      } else {
       childHistogramHeight = (int) (childData.getValue() / childMaxValueArray.get(i) * maxHistogramHeight);
      }
      histogramPaintRect.top = height - childHistogramHeight - coordinateAxisWidth - distanceFormGroupNameToAxis - groupNameTextSize;
      histogramPaintRect.bottom = histogramPaintRect.top + childHistogramHeight;
      int[] histogramShaderColor = histogramShaderColorArray.get(i);
      LinearGradient shader = null;
      if (histogramShaderColor != null && histogramShaderColor.length > 0) {
       shader = getHistogramShader(histogramPaintRect.left, chartPaddingTop + distanceFromValueToHistogram + histogramValueTextSize,
         histogramPaintRect.right, histogramPaintRect.bottom, histogramShaderColor);
      }
      histogramPaint.setShader(shader);
      canvas.drawRect(histogramPaintRect, histogramPaint);
      String childHistogramHeightValue = StringUtil.NumericScaleByFloor(String.valueOf(childData.getValue()), histogramValueDecimalCount) + childData.getSuffix();

      float valueTextX = xAxisOffset + (histogramHistogramWidth - histogramValuePaint.measureText(childHistogramHeightValue)) / 2;
      // 数值绘制Y轴位置特别处理
      float valueTextY = histogramPaintRect.top - distanceFormGroupNameToAxis + (histogramValueFontMetrics.bottom) / 2;
      canvas.drawText(childHistogramHeightValue, valueTextX, valueTextY, histogramValuePaint);
      int deltaX = i < childDataList.size() - 1 ? histogramHistogramWidth + histogramInterval : histogramHistogramWidth;
      groupWidth += deltaX;
      // 注意此处偏移量累加
      xAxisOffset += i == childDataList.size() - 1 ? deltaX + groupInterval : deltaX;
     }
     String groupName = groupData.getGroupName();
     float groupNameTextWidth = groupNamePaint.measureText(groupName);
     float groupNameTextX = xAxisOffset - groupWidth - groupInterval + (groupWidth - groupNameTextWidth) / 2;
     // 组名绘制Y轴位置特别处理
     float groupNameTextY = (height - groupNameFontMetrics.bottom / 2);
     canvas.drawText(groupName, groupNameTextX, groupNameTextY, groupNamePaint);
    }
   }
  }
 }

 private LinearGradient getHistogramShader(float x0, float y0, float x1, float y1, int[] colors) {
  return new LinearGradient(x0, y0, x1, y1, colors, null, Shader.TileMode.CLAMP);
 }
}

代码就这一点,阅读起来应该不难,如有疑问欢迎留言

自定义属性如下:

 <declare-styleable name="MultiGroupHistogramView">
  <attr name="coordinateAxisWidth" format="dimension" />
  <attr name="coordinateAxisColor" format="color" />
  <attr name="groupNameTextColor" format="color" />
  <attr name="groupNameTextSize" format="dimension" />
  <attr name="groupInterval" format="dimension" />
  <attr name="histogramInterval" format="dimension" />
  <attr name="histogramValueTextColor" format="color" />
  <attr name="histogramValueTextSize" format="dimension" />
  <attr name="histogramHistogramWidth" format="dimension" />
  <attr name="histogramPaddingStart" format="dimension" />
  <attr name="histogramPaddingEnd" format="dimension" />
  <attr name="chartPaddingTop" format="dimension" />
  <attr name="distanceFormGroupNameToAxis" format="dimension" />
  <attr name="distanceFromValueToHistogram" format="dimension" />
  <!--图表数值小数点位数-->
  <attr name="histogramValueDecimalCount">
   <enum name="ZERO" value="0" />
   <enum name="ONE" value="1" />
   <enum name="TWO" value="2" />
  </attr>
 </declare-styleable>

下面贴出使用方法:

 private void initMultiGroupHistogramView() {
  Random random = new Random();
  int groupSize = random.nextInt(5) + 10;
  List<MultiGroupHistogramGroupData> groupDataList = new ArrayList<>();
  // 生成测试数据
  for (int i = 0; i < groupSize; i++) {
   List<MultiGroupHistogramChildData> childDataList = new ArrayList<>();
   MultiGroupHistogramGroupData groupData = new MultiGroupHistogramGroupData();
   groupData.setGroupName("第" + (i + 1) + "组");
   MultiGroupHistogramChildData childData1 = new MultiGroupHistogramChildData();
   childData1.setSuffix("分");
   childData1.setValue(random.nextInt(50) + 51);
   childDataList.add(childData1);

   MultiGroupHistogramChildData childData2 = new MultiGroupHistogramChildData();
   childData2.setSuffix("%");
   childData2.setValue(random.nextInt(50) + 51);
   childDataList.add(childData2);
   groupData.setChildDataList(childDataList);
   groupDataList.add(groupData);
  }
  multiGroupHistogramView.setDataList(groupDataList);
  int[] color1 = new int[]{getResources().getColor(R.color.color_orange), getResources().getColor(R.color.colorPrimary)};

  int[] color2 = new int[]{getResources().getColor(R.color.color_supper_tip_normal), getResources().getColor(R.color.bg_supper_selected)};
  // 设置直方图颜色
  multiGroupHistogramView.setHistogramColor(color1, color2);
 }

完整示例:https://github.com/670832188/TestApp (本地下载)

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Android实现简易的柱状图和曲线图表实例代码

    前言 之前有写过一个图表lib,但是开发的速度,大多很难跟上产品需求变化的脚步,所以修改了下原先的图表库,支持图表下面能整合table显示对应的类目,用曲线替换了折线,支持多曲线的显示,增加了显示的动画,增加了一些可定制的属性,支持水平柱状图和叠加柱状图,以及多曲线图和饼状图的显示,下面话不多说了,来一起看看详细的介绍吧. 1.效果图 2.各种图表的使用方式 1.饼状图 这个和原先的使用一样,只不过增加了一个动画,可以参看之前的文章,饼状图使用. 2.水平多柱状图 2.1 xml布局 <well

  • 详解Android自定义View--自定义柱状图

    绪论 转眼间,2016伴随着互联网寒冬和帝都的雾霾马上就过去了,不知道大家今年一整年过得怎么样?最近票圈被各个城市的雾霾刷屏,内心难免会动荡,庆幸自己早出来一年,也担忧着自己的未来的职业规划.无所谓了,既然选择了这个行业,我觉得大家就应该坚持下去,路是自己走的,及时再寒冬,只要你足够优秀,足够努力,相信你最后还是会找到自己满意的工作的.最后还要感谢今年博客之星大家对我的投票支持,非常感谢.不多说了,今天的主题是它–对,自定义View柱状图. 先来说说我最近在做什么吧?好久没有写博客了,最近手里有

  • Android自定义带增长动画和点击弹窗提示效果的柱状图DEMO

    项目中最近用到各种图表,本来打算用第三方的,例如MPAndroid,这是一个十分强大的图表库,应用起来十分方便,但是最终发现和设计不太一样,没办法,只能自己写了.今天将写好的柱状图的demo贴在这,该柱状图可根据数据的功能有一下几点: 1. 根据数据的多少,动态的绘制柱状图柱子的条数: 2. 柱状图每条柱子的绘制都有动态的动画效果: 3. 每条柱子有点击事件,点击时弹出提示框,显示相关信息,规定时间后,弹窗自动消失. 好了,先上演示图: 下边贴出相关代码: 自定义柱状图类: package co

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

    为何要用带动画的柱状图呢? 最近,项目中遇到一个地方,要用到柱状图.所以这篇文章主要讲怎么搞一个柱子.100行代码,搞定柱状图! 圆角,头顶带数字.恩,这样用drawable也可以搞定.但是,这个柱子是有一个动画的,就是进入到界面的时候柱子不断的长高.这样的话,综合考虑还是用自定义View来做比较简便.效果如下图了: 完整Demo地址请到我的github下载地址: https://github.com/lixiaodaoaaa/ColumnAnimViewProject 关于尺寸 控件尺寸直接来

  • Android自定义view实现动态柱状图

    先看一下动态柱状图效果. 主要功能是可以自定义指定的字体,柱状图颜色,动态效果. 在自定义view public class Histogram extends View { int MAX = 100;//矩形显示的最大值 int corner = 0; //矩形的角度. 设置为0 则没有角度. double data = 0.0;//显示的数 double tempData = 0; //初始数据 int textPadding = 50; //字体与矩形图的距离 Paint mPaint;

  • MPAndroidChart开源图表库的使用介绍之饼状图、折线图和柱状图

    MPAndroidChart开源图表库之饼状图 为大家介绍一款图标开源库MPAndroidChart,它不仅可以在Android设备上绘制各种统计图表,而且可以对图表进行拖动和缩放操作,用起来非常灵活.MPAndroidChart同样拥有常用的图表类型:线型图.饼图.柱状图和散点图. mpandroidchartlibrary.jar包下载地址: https://github.com/PhilJay/MPAndroidChart/releases 下面主要实现以下饼状图: 1.从上面的地址中下载

  • Android自定义柱状图表的方法实例

    前言 本文将通过示例代码介绍如何自定义简单的直方图表,此图表并非常见的直方图表,而是可以分组的.此文不会过多涉及原理,比较简单,示例图片如下(gif图片没有制作好,有闪烁,请见谅): 对于该示例的代码实现,其实重点在于坐标轴.文字.直方图的位置控制,需要随滑动距离而动态更新.注意事项会在示例代码中标注.下面贴出示例代码 public class MultiGroupHistogramView extends View { private int width; private int height

  • Android自定义View的实现方法实例详解

    一.自绘控件 下面我们准备来自定义一个计数器View,这个View可以响应用户的点击事件,并自动记录一共点击了多少次.新建一个CounterView继承自View,代码如下所示: 可以看到,首先我们在CounterView的构造函数中初始化了一些数据,并给这个View的本身注册了点击事件,这样当CounterView被点击的时候,onClick()方法就会得到调用.而onClick()方法中的逻辑就更加简单了,只是对mCount这个计数器加1,然后调用invalidate()方法.通过 Andr

  • Android自定义验证码输入框的方法实例

    目录 实践过程 总结 实践过程 前面我们学完了EditText和TextView两个组件,但是,光学不练没意思. 所以今天我们趁热打铁,利用两个组件实现个自定义验证码输入框. 思路前瞻: 隐形EditText接收输入,显性TextView展示内容 时刻监听EditText回调更改内容 自定义RelativeLayout 布局代码: <?xml version="1.0" encoding="utf-8"?><!--自定义验证码View-->

  • Android自定义水波纹动画Layout实例代码

    话不多说,我们先来看看效果: Hi前辈搜索预览 这一张是<Hi前辈>的搜索预览图,你可以在这里下载这个APP查看更多效果: http://www.wandoujia.com/apps/com.superlity.hiqianbei LSearchView 这是一个MD风格的搜索框,集成了ripple动画以及search时的loading,使用很简单,如果你也需要这样的搜索控件不妨来试试:https://github.com/onlynight/LSearchView RippleEverywh

  • Android自定义dialog简单实现方法

    本文实例讲述了Android自定义dialog简单实现方法.分享给大家供大家参考,具体如下: @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.function_music); // 实例化新的窗口 Window w = getWindow(); // 获取默认显示数据 Display display

  • Android自定义view实现圆环效果实例代码

    先上效果图,如果大家感觉不错,请参考实现代码.           重要的是如何实现自定义的view效果 (1)创建类,继承view,重写onDraw和onMesure方法 public class CirclePercentBar extends View{ private Context mContext; private int mArcColor; private int mArcWidth; private int mCenterTextColor; private int mCent

  • Android 自定义imageview实现图片缩放实例详解

    Android 自定义imageview实现图片缩放实例详解 觉得这个自定义的imageview很好用 性能不错  所以拿出来分享给大家  因为不会做gif图  所以项目效果 就不好贴出来了  把代码贴出来 1.项目结构图 2.Compat.class package com.suo.image; import android.os.Build.VERSION; import android.os.Build.VERSION_CODES; import android.view.View; pu

  • Android自定义view实现太极效果实例代码

    Android自定义view实现太极效果实例代码 之前一直想要个加载的loading.却不知道用什么好,然后就想到了太极图标,最后效果是有了,不过感觉用来做loading简直丑到爆!!! 实现效果很简单,我们不要用什么贝塞尔曲线啥的,因为太极无非就是圆圆圆,只要画圆就ok了.来上代码: 因为有黑有白,所以定义2个画笔分别为黑和白. private void inital() { whitePaint = new Paint(); whitePaint.setAntiAlias(true); wh

  • Android连接指定Wifi的方法实例代码

    本篇文章主要记录一下Android中打开Wifi.获取Wifi接入点信息及连接指接入点的方法. 自己写的demo主要用于测试接口的基本功能,因此界面及底层逻辑比较粗糙. demo的整体界面如下所示: 上图中的OPEN按键负责开启Wifi: GET按键负责获取扫描到的接入点信息. 当获取到接入点信息后,我选取了其中的名称及信号强度,以列表的形式显示在主界面下方,如下图: 当点击列表中的Item时,就会去连接对应的接入点. 自己的逻辑比较简单,测试时的代码,假定连接的是不许要密码或密码已知的接入点.

  • Android 自定义阴影效果详解及实例

    Android 自定义阴影效果详解及实例 Android5.X中,Google为其增加了两个属性 android:elevation=" " 与 android:translationZ=" ",对应垂直方向上的高度变化.系统会自动增加阴影效果. 在TabLayout中增加android:elevation=" 8dp" ,效果如下: 箭头指向的就是系统为我们默认提供,结果差强人意.那我们是不是可以自定义阴影,不使用系统提供的. 自定义阴影效果

随机推荐