Android自定义View(二)
前言
魅族手机的闹钟应用中有个倒计时,这个控件还是蛮有趣的。左边是魅族闹钟,右边是我们最终实现的效果,虽然有些细节还需优化,不过基本上已经达到了想要的效果,我们先来就来看看如何实现吧。
分析
确定宽高
对一个Android自定义控件来说,一般都经过三个步骤
onLayout()
onMeasure()
onDraw()
onLayout明确子控件在父控件中的位置(本控件不需要重写),onMeasure是确定控件的大小(宽、高),而onDraw是我们重点关注的方法,我们需要在这个方法中写入显示View的逻辑代码。
对于本控件,控件的高度 应该等于细线的高度(mLineHeight)加上数字的高度(mFontHeight),当然为了好看,中间需要设上一些边距(mPadding),因此本控件的高度应该为 mFontHeight + mLineHeight + 10 + mPadding,测量代码如下
private int measureHeight(int heightMeasureSpec) {
int mode = MeasureSpec.getMode(heightMeasureSpec);
int size = MeasureSpec.getSize(heightMeasureSpec);
switch (mode) {
case MeasureSpec.EXACTLY:
return size ;
case MeasureSpec.AT_MOST:
return Math.min(size, mFontHeight + mLineHeight + 10 + mPadding) ;
}
return size ;
}
同样地,控件的宽度其实就是0~1000的间隔,测量代码如下
private int measureWidth(int widthMeasureSpec) {
int mode = MeasureSpec.getMode(widthMeasureSpec);
int size = MeasureSpec.getSize(widthMeasureSpec);
switch (mode) {
case MeasureSpec.EXACTLY:
return size ;
case MeasureSpec.AT_MOST:
int result = getPaddingLeft() + mContentWidth + getPaddingRight() ;
return Math.min(size, result) ;
}
return size ;
}
画刻度尺
重点在于刻度尺的计算。思路是先draw上头的数字,然后再draw下边的线条,判断位置确定是否需要draw上头的数字即可。其实就是坐标的计算。代码如下
int startX = mPadding;
int stopX = mPadding;
int stopY =mHeight - mPadding;
for (int i = 0 ; i<=mContentWidth ; i += mFontWidth)
if (i % (mFontWidth *10) == 0) {
canvas.drawLine(startX + i, mFontHeight + mPadding + 5 , stopX + i, stopY, mTextPaint);
canvas.drawText(i + "", startX + i, mFontHeight + mPadding, mTextPaint);
} else if (i % mFontWidth == 0) {
canvas.drawLine(startX + i, mFontHeight + mPadding + 10, stopX + i, stopY, mPaint);
}
让View动起来
Android本身提供了移动View的API,因此让View动起来也是不难的。两种思路
监听Touch事件,当Touch坐标变化时,计算坐标位置,不断调用scrollTo(x,0)达到变换坐标的目的
监听Touch事件,记录上次横坐标和本次横坐标的差值,然后调用scrollBy(delta, 0) 即可移动
其实两种方法本质上都是一样的。ScrollBy其实也是调用了scrollTo方法。本文采用方法二。其代码如下
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
mLastX = (int) event.getX();
break;
case MotionEvent.ACTION_MOVE:
int x = (int) event.getX();
int deltaX = x - mLastX;
scrollBy(-deltaX, 0);
mLastX = x;
break;
}
return true;
}
当然了,我们是不能让View无限移动的,因此需要重写scrollBy方法,限制View不能超过边界。
代码如下
@Override
public void scrollBy(int x, int y) {
super.scrollBy(x, y);
if (x < 0) { // drag to left
if (getScrollX() < -getCenter() + mPadding) {
scrollTo(-getCenter() + mPadding, 0);
}
} else
if (x >0) { // drag to right
if (mContentWidth - getScrollX() + x < getCenter()) {
scrollTo(mContentWidth - getCenter() + mFontWidth, 0);
}
}
}
当超过边界时,直接调用scrollTo,让View停留在特定的位置即可。需要注意的一点是,View往左滑动时,ScrollX的值是负的。
完整代码
package com.nancyyihao.demo;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Scroller;
public class RulerView extends View {
private static final String TAG = RulerView.class.getSimpleName() ;
private TextPaint mTextPaint;
private Paint mPaint ;
private int mWidth;
private int mHeight;
private int mPadding = 10;
private Scroller mScroller ;
private int mLastX;
private int mContentWidth = 1000;
private int mLineHeight = 50;
private int mFontHeight;
private int mFontWidth = 10;
private onValueChangedListener mValueChangedListener;
public interface onValueChangedListener {
void onValueChanged(int newValue);
}
public RulerView(Context context) {
super(context);
init(context);
}
public RulerView(Context context, AttributeSet attrs) {
super(context, attrs);
init(context);
}
public RulerView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init(context);
}
public void setOnValueChangedListener(onValueChangedListener listener) {
this.mValueChangedListener = listener ;
}
private void init(Context context) {
mTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG | Paint.DITHER_FLAG | Paint.LINEAR_TEXT_FLAG);
mTextPaint.setTextAlign(Paint.Align.CENTER);
mTextPaint.setColor(Color.parseColor("#FF4081"));
mTextPaint.setTextSize(30);
mTextPaint.setStrokeWidth(2f);
Paint.FontMetrics fontMetrics = mTextPaint.getFontMetrics();
mFontHeight = Math.round(Math.abs(fontMetrics.top) + Math.abs(fontMetrics.bottom)) ;
mPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.DITHER_FLAG);
mPaint.setColor(Color.DKGRAY);
mPaint.setStrokeWidth(2f);
mPaint.setTextSize(30);
mPaint.setTextAlign(Paint.Align.CENTER);
//mScroller = new Scroller(context) ;
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
mWidth = measureWidth(widthMeasureSpec);
mHeight = measureHeight(heightMeasureSpec);
setMeasuredDimension(mWidth , mHeight );
}
private int measureWidth(int widthMeasureSpec) {
int mode = MeasureSpec.getMode(widthMeasureSpec);
int size = MeasureSpec.getSize(widthMeasureSpec);
switch (mode) {
case MeasureSpec.EXACTLY:
return size ;
case MeasureSpec.AT_MOST:
int result = getPaddingLeft() + mContentWidth + getPaddingRight() ;
return Math.min(size, result) ;
}
return size ;
}
private int measureHeight(int heightMeasureSpec) {
int mode = MeasureSpec.getMode(heightMeasureSpec);
int size = MeasureSpec.getSize(heightMeasureSpec);
switch (mode) {
case MeasureSpec.EXACTLY:
return size ;
case MeasureSpec.AT_MOST:
return Math.min(size, mFontHeight + mLineHeight + 10 + mPadding) ;
}
return size ;
}
// private void smoothScrollTo(int destX, int destY) {
// int scrollX = getScrollX() ;
// int delta = destX - scrollX ;
// mScroller.startScroll(scrollX, 0, delta, 0 , 1000);
// invalidate();
// }
//
// @Override
// public void computeScroll() {
// super.computeScroll();
// if (mScroller.computeScrollOffset()) {
// smoothScrollTo(mScroller.getCurrX(), mScroller.getCurrY());
// postInvalidate();
//// ((View) getParent()).scrollTo(mScroller.getCurrX(),mScroller.getCurrY());
//// invalidate();
// }
// }
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
int startX = mPadding;
int stopX = mPadding;
int stopY =mHeight - mPadding;
for (int i = 0 ; i<=mContentWidth ; i += mFontWidth)
if (i % (mFontWidth *10) == 0) {
canvas.drawLine(startX + i, mFontHeight + mPadding + 5 , stopX + i, stopY, mTextPaint);
canvas.drawText(i + "", startX + i, mFontHeight + mPadding, mTextPaint);
} else if (i % mFontWidth == 0) {
canvas.drawLine(startX + i, mFontHeight + mPadding + 10, stopX + i, stopY, mPaint);
}
}
private int calcValue() {
return ( getCenter() + getScrollX() - mPadding) ; //minus startX
}
private int getCenter() {
return (getRight() - getLeft()) / 2 ;
}
@Override
public void scrollBy(int x, int y) {
super.scrollBy(x, y);
if (x < 0) { // drag to left
if (getScrollX() < -getCenter() + mPadding) {
scrollTo(-getCenter() + mPadding, 0);
}
} else
if (x >0) { // drag to right
if (mContentWidth - getScrollX() + x < getCenter()) {
scrollTo(mContentWidth - getCenter() + mFontWidth, 0);
}
}
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
mLastX = (int) event.getX();
break;
case MotionEvent.ACTION_MOVE:
int x = (int) event.getX();
int deltaX = x - mLastX;
scrollBy(-deltaX, 0);
mLastX = x;
if (mValueChangedListener != null)
mValueChangedListener.onValueChanged(calcValue());
break;
}
return true;
}
}
总结
整体上还是比较粗糙,原形虽然有了,但是还需要优化。
参考
【Android】自定义View —— 滑动的次数选择器
android 滚轮刻度尺的实现
Android View自定义专题二(View滑动的实现)
Android自定义View(二)的更多相关文章
- Android自定义View (二) 进阶
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24300125 继续自定义View之旅,前面已经介绍过一个自定义View的基础的例 ...
- Android 自定义View (二) 进阶
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24300125 继续自定义View之旅,前面已经介绍过一个自定义View的基础的例 ...
- Android 自定义View二(深入了解自定义属性attrs.xml)
1.为什么要自定义属性 要使用属性,首先这个属性应该存在,所以如果我们要使用自己的属性,必须要先把他定义出来才能使用.但我们平时在写布局文件的时候好像没有自己定义属性,但我们照样可以用很多属性,这是为 ...
- Android 自定义view(二) —— attr 使用
前言: attr 在前一篇文章<Android 自定义view -- attr理解>已经简单的进行了介绍和创建,那么这篇文章就来一步步说说attr的简单使用吧 自定义view简单实现步骤 ...
- Android 自定义View及其在布局文件中的使用示例(二)
转载请注明出处 http://www.cnblogs.com/crashmaker/p/3530213.html From crash_coder linguowu linguowu0622@gami ...
- Android自定义View(二、深入解析自定义属性)
转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51468648 本文出自:[openXu的博客] 目录: 为什么要自定义属性 怎样自定义属性 ...
- Android 自定义 view(三)—— onDraw 方法理解
前言: 上一篇已经介绍了用自己定义的属性怎么简单定义一个view<Android 自定义view(二) -- attr 使用>,那么接下来我们继续深究自定义view,下一步将要去简单理解自 ...
- Android 自定义View及其在布局文件中的使用示例(三):结合Android 4.4.2_r1源码分析onMeasure过程
转载请注明出处 http://www.cnblogs.com/crashmaker/p/3549365.html From crash_coder linguowu linguowu0622@gami ...
- android自定义View之仿通讯录侧边栏滑动,实现A-Z字母检索
我们的手机通讯录一般都有这样的效果,如下图: OK,这种效果大家都见得多了,基本上所有的android手机通讯录都有这样的效果.那我们今天就来看看这个效果该怎么实现. 一.概述 1.页面功能分析 整体 ...
- Android自定义View研究--View中的原点坐标和XML中布局自定义View时View触摸原点问题
这里只做个汇总~.~独一无二 文章出处:http://blog.csdn.net/djy1992/article/details/9715047 Android自定义View研究--View中的原点坐 ...
随机推荐
- 七、vue计算属性
细节流程图 初始化 计算属性的初始化是发生在 Vue 实例初始化阶段的 initState 函数中,执行了 if (opts.computed) initComputed(vm, opts.compu ...
- GCD 开发详情
目录 一.简介 二.dispatch Queue - 队列 三.dispatch Groups - 组 四.dispatch Semaphores - 信号量 五.dispatch Barriers ...
- hdu 2510 符号三角形 (DFS+打表)
符号三角形 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submi ...
- [codeforces934D]A Determined Cleanup
[codeforces934D]A Determined Cleanup 试题描述 In order to put away old things and welcome a fresh new ye ...
- [2018-9-4T2]探索黑暗dark
题目大意:有一棵树,第$i$个点的点权为$s_i(s_1>0)$.初始有每个点都是亮的.$m$次修改,每次改变一个点的亮暗,回答包含$1$的全亮的连通块中点权和最大的连通块的和的值. 题解:正解 ...
- angular组件--tips提示功能
将组件封装起来在项目中开发很实用,之前遭遇过一次痛苦的经历,那阵子改的要吐血了.常用的组件封装起来,改公共的地方,往往多处受用. 例如:我在项目中引用 tips.text('加载中...',fals ...
- tortoise git使用 git版本库的rsa key来进行ssh连接
接触git以来 ,开始时用了命令行,但是命令行总归不如图形化菜单方便明了,而GIT本身自带的GUI又用的不习惯,以前用过许久的TOTORISE SVN,幸好有TORTOISE GIT,这个版本图形化工 ...
- error LNK2001: 无法解析的外部符号 "public: virtual long __stdcall CBaseFilter(转)
原文转自 https://www.cnblogs.com/xiongjiaji/archive/2010/12/31/2476565.html 今天用VS2005编译DirectShow程序,发现出来 ...
- Linux内核实践之tasklet机制【转】
转自:http://blog.csdn.net/bullbat/article/details/7423321 版权声明:本文为博主原创文章,未经博主允许不得转载. 作者:bullbat 源代码分析与 ...
- android studio 无法调试debug,(能运行安装)
请检查清单文件的改为true就可以调试了 android:debuggable="true"