在有些Android应用中,为了方便快速定位,经常会看到屏幕右侧有一个字母索引表,今天尝试使用自定义View的方式实现了索引表的基本布局。

  字母索引表的样式如下面的示意图所示,

  此时我们至少需要知道以下几个参数值:1.字母大小;2.单个字母所在区域的宽度;3.单个字母所在区域的高度。现在看如何实现:

    /**
* 26个英文字母以及一个#字符,#字符是为了索引非英文字母的内容,比如电话号码。
*/
private String[] mAlphabetTable = {
"A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K",
"L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W",
"X", "Y", "Z", "#"
};
/**
* 单个字母的大小
*/
private int mAlphabetSize = 0;
/**
* 字母表的宽度
*/
private int mWidth = 0;
/**
* 字母表的高度
*/
private int mHeight = 0;
/**
* 窗口高度
*/
private int mDisplayHeight;
/**
* 单个字母所在区域的高度
*/
private int mStepPixel; private int mFastScrollViewHeight = 0;
private Rect[] mAlphabetRect = new Rect[27];
private Rect mPointRectAll = new Rect();
private static int mAlphabetLeftPadding = 0;
private boolean[] mPoint = new boolean[27]; public AlphabetFastScorll(Context context) {
this(context, null);
} public AlphabetFastScorll(Context context, AttributeSet attrs) {
super(context, attrs);
this.setAlpha(1.0f);//设置透明度1
DisplayMetrics metric = new DisplayMetrics();
((WindowManager) getContext().getSystemService(getContext().WINDOW_SERVICE)).getDefaultDisplay().getMetrics(metric);
mDisplayHeight = metric.heightPixels;//获取窗口高度
}

  

  在构造函数里,只设置了索引表的透明度和初始化mDisplayHeight参数。索引表的绘制工作大部分是在onDraw()里完成。

    @Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
} @Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
mStepPixel = getContext().getResources().getDimensionPixelOffset(R.dimen.fast_scroll_size);
mFastScrollViewHeight = mStepPixel * 27;
     
//初始化27个字母每个字母对应所在区域的大小范围
InitAlphabetRect(); //初始化字母索引表的背景大小
initPointRect(mFastScrollViewHeight, mStepPixel); //设置背景颜色
Paint paintB = new Paint();
paintB.setColor(ContextCompat.getColor(getContext(), R.color.colorAccent));
//paintB.setAlpha(0);
canvas.drawRect(mPointRectAll, paintB);
     //设置字母中大小,颜色,位置等
Paint paint = new Paint();
float textSize = getContext().getResources().getDimensionPixelOffset(R.dimen.asus_index_size);
paint.setColor(ContextCompat.getColor(getContext(), R.color.fast_scroll_text_color));
paint.setTextAlign(Align.CENTER);
paint.setTextSize(textSize);
paint.setAntiAlias(true); for (int i = 0; i < mAlphabetRect.length; i++) {
int x = (mAlphabetRect[i].left + mAlphabetRect[i].right) / 2;
int y = mAlphabetRect[i].bottom; y = y - (int) (0.5 * (mStepPixel - textSize));
//字母绘制在mAlphabetRect[i]区域中间
canvas.drawText(mAlphabetTable[i], x, y, paint);
}
} private void InitAlphabetRect() {
for (int i = 0; i < mAlphabetRect.length; i++) {
mAlphabetRect[i] = new Rect();
mAlphabetRect[i].left = mAlphabetLeftPadding;
mAlphabetRect[i].right = mAlphabetLeftPadding + this.getWidth();
//Log.d(TAG,"InitAlphabetRect right = " + mAlphabetRect[i].right);
if (i == 0) {
mAlphabetRect[i].top = 0;
mAlphabetRect[i].bottom = mStepPixel;
} else {
mAlphabetRect[i].top = mAlphabetRect[i-1].top + mStepPixel;
mAlphabetRect[i].bottom = mAlphabetRect[i].top + mStepPixel;
}
}
} private void initPointRect(int fastScrollViewHeight, int stepPixel) {
int canvasWidth = getContext().getResources().getDimensionPixelOffset(R.dimen.asus_index_canvas_width); mPointRectAll = new Rect();
mPointRectAll.left = 0;
mPointRectAll.right = canvasWidth;
mPointRectAll.top = 0;
mPointRectAll.bottom = this.getHeight();
}

  在onDraw()里做了以下几件事情:

  1,初始化索引表字母所在区域的范围:新建一个Rect对象,设置其左右和上下大小。宽度大小由布局文件中提供,高度为mStepPixel。

  2,初始化索引表背景范围:宽 = asus_index_canvas_width,高 = this.getHeight();

  3,已经知道每个字母所在区域范围以及索引背景范围,新建两个Paint在Canves上进行绘制;

  通过以上几步就可以简单实现索引表控件,光光显示一个索引表肯定没啥意义。一般都是将索引表与ListView组合,实现快速定位功能。下篇准备模仿联系人应用,实现通过索引表快速定位到联系人。

Android 自定义View-字母索引表(一)的更多相关文章

  1. 转载:android自定义view实战(温度控制表)!

    效果图 package cn.ljuns.temperature.view; import com.example.mvp.R; import android.content.Context;impo ...

  2. Android为TV端助力 转载:android自定义view实战(温度控制表)!

    效果图 package cn.ljuns.temperature.view; import com.example.mvp.R; import android.content.Context;impo ...

  3. android自定义View之仿通讯录侧边栏滑动,实现A-Z字母检索

    我们的手机通讯录一般都有这样的效果,如下图: OK,这种效果大家都见得多了,基本上所有的android手机通讯录都有这样的效果.那我们今天就来看看这个效果该怎么实现. 一.概述 1.页面功能分析 整体 ...

  4. Android自定义View之ProgressBar出场记

    关于自定义View,我们前面已经有三篇文章在介绍了,如果筒子们还没阅读,建议先看一下,分别是android自定义View之钟表诞生记.android自定义View之仿通讯录侧边栏滑动,实现A-Z字母检 ...

  5. android自定义View之NotePad出鞘记

    现在我们的手机上基本都会有一个记事本,用起来倒也还算方便,记事本这种东东,如果我想要自己实现,该怎么做呢?今天我们就通过自定义View的方式来自定义一个记事本.OK,废话不多说,先来看看效果图. 整个 ...

  6. Android自定义View(CustomCalendar-定制日历控件)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/54020386 本文出自:[openXu的博客] 目录: 1分析 2自定义属性 3onMeas ...

  7. Android自定义View(三、深入解析控件测量onMeasure)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51490283 本文出自:[openXu的博客] 目录: onMeasure什么时候会被调用 ...

  8. Android自定义View(二、深入解析自定义属性)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51468648 本文出自:[openXu的博客] 目录: 为什么要自定义属性 怎样自定义属性 ...

  9. 简单说说Android自定义view学习推荐的方式

    这几天比较受关注,挺开心的,嘿嘿. 这里给大家总结一下学习自定义view的一些技巧.  以后写自定义view可能不会写博客了,但是可以开源的我会把源码丢到github上我的地址:https://git ...

随机推荐

  1. 解决Qt程序在Linux下无法输入中文的办法(与下文连接)

    在安装QT集成开发工具包之前需要先安装build-essential和libncurses5-dev这两个开发工具和库,libncurses5-dev库是一个在Linux/Unix下广泛应用的图形函数 ...

  2. Spring学习笔记--构造器注入

    之前讲到的名为"duke"的bean有一个私有成员变量beanBags代表这个杂技师bean的一次性能够抛出的最多的数量,Juggler有一个构造函数,构造函数的第一个参数(这里只 ...

  3. 关于Ethread的一些研究

    环境 win764 以TP为例 ring3保护 它会在windbg断下 这个时候我们需要拿到当前线程对象 应该到 当前使用的CPU的地址 _KPRCB-> CurrentThread 就是当前线 ...

  4. MUI 页面跳转(传值+接收)

    官方:做web app,一个无法避开的问题就是转场动画:web是基于链接构建的,从一个页面点击链接跳转到另一个页面, 如果通过有刷新的打开方式,用户要面对一个空白的页面等待: 如果通过无刷新的方式,用 ...

  5. Android之ListView分页数据加载

    1.效果如下: 实例如下:  上图的添加数据按钮可以换成一个进度条  因为没有数据所以我加了一个按钮添加到数据库用于测试:一般在服务器拉去数据需要一定的时间,所以可以弄个进度条来提示用户: 点击加载按 ...

  6. 【BZOJ1879】[Sdoi2009]Bill的挑战 状压DP

    [BZOJ1879][Sdoi2009]Bill的挑战 Description Input 本题包含多组数据.  第一行:一个整数T,表示数据的个数.  对于每组数据:  第一行:两个整数,N和K(含 ...

  7. 'Settings' object has no attribute 'TEMPLATE_DEBUG' 的解决方法

    找到该Django项目下的settings文件,把 DEBUG = True 改为 DEBUG = False 就可以正常浏览显示了 参考:https://stackoverflow.com/ques ...

  8. Java虚拟机规范----JVM体系结构

    一.Java平台的结构图 二.JVM与JRE.JDK关系? JVM:Java Virtual Machine(Java虚拟机),负责执行符合规范的Class文件 JRE:Java Runtime En ...

  9. Windows下Git免密码pull&push

    Windows下Git在使用http方式的时候clone,pull,push需要输入用户名及密码,通过以下设置可以免密码 在用户文件夹创建文件.git-credentials内容如下 https:// ...

  10. PAT 1040

    字符串APPAPT中包含了两个单词"PAT",其中第一个PAT是第2位(P),第4位(A),第6位(T):第二个PAT是第3位(P),第4位(A),第6位(T). 现给定字符串,问 ...