常用的自定义控件四(QuickBarView)

  1.          `// 获取文本的高度`
    
  2.          `Rect bounds = new Rect();// 矩形`
    
  3.          `mPaint.getTextBounds(text, 0, text.length(), bounds);`
    
  4.          `int textHeight = bounds.height();`
    
  5.          `int y = (int) (cellHeight / 2.0f + textHeight / 2.0f + i * cellHeight);`
    
  6.          `// 根据按下的字母, 设置画笔颜色`
    
  7.          `mPaint.setColor(mLastTouchIndex == i ? mSelectColor : mNormalColor);`
    
  8.          `// 绘制文本A-Z`
    
  9.          `canvas.drawText(text, x, y, mPaint);`
    
  10.      `}`
    
  11.  `}`
    
  12.  `@Override`
    
  13.  `public boolean onTouchEvent(MotionEvent event) {`
    
  14.      `int index = -1;`
    
  15.      `switch (MotionEventCompat.getActionMasked(event)) {`
    
  16.          `case MotionEvent.ACTION_DOWN:`
    
  17.              `// 获取当前触摸到的字母索引`
    
  18.              `index = (int) (event.getY() / cellHeight);`
    
  19.              `if (index >= 0 && index < LETTERS.length) {`
    
  20.                  `// 判断是否跟上一次触摸到的一样,不一样才进行回调`
    
  21.                  `if (index != mLastTouchIndex) {`
    
  22.                      `if (listener != null) {`
    
  23.                          `//`
    
  24.                          `listener.onLetterUpdate(LETTERS[index]);`
    
  25.                      `}`
    
  26.                      `Log.d(TAG, "onTouchEvent: " + LETTERS[index]);`
    
  27.                      `//记录上一次触摸的Index为当前的index;`
    
  28.                      `mLastTouchIndex = index;`
    
  29.                  `}`
    
  30.              `}`
    
  31.              `break;`
    
  32.          `case MotionEvent.ACTION_MOVE:`
    
  33.              `index = (int) (event.getY() / cellHeight);`
    
  34.              `if (index >= 0 && index < LETTERS.length) {`
    
  35.                  `// 判断是否跟上一次触摸到的一样`
    
  36.                  `if (index != mLastTouchIndex) {`
    
  37.                      `if (listener != null) {`
    
  38.                          `listener.onLetterUpdate(LETTERS[index]);`
    
  39.                      `}`
    
  40.                      `Log.d(TAG, "onTouchEvent: " + LETTERS[index]);`
    
  41.                      `mLastTouchIndex = index;`
    
  42.                  `}`
    
  43.              `}`
    
  44.              `break;`
    
  45.          `case MotionEvent.ACTION_UP:`
    
  46.              `// 手指抬起的时候重置`
    
  47.              `mLastTouchIndex = -1;`
    
  48.              `break;`
    
  49.          `default:`
    
  50.              `break;`
    
  51.      `}`
    
  52.      `//调用这个方法会重新调用draw方法,重新绘制`
    
  53.      `invalidate();`
    
  54.      `return true;`
    
  55.  `}`
    
  56.  `/**`
    
  57.   `* 当大小 改变的时候会回调这个方法,`
    
  58.   `* 这里我们就不主动调用measure()方法了`
    
  59.   `*`
    
  60.   `* @param w`
    
  61.   `* @param h`
    
  62.   `* @param oldw`
    
  63.   `* @param oldh`
    
  64.   `*/`
    
  65.  `@Override`
    
  66.  `protected void onSizeChanged(int w, int h, int oldw, int oldh) {`
    
  67.      `super.onSizeChanged(w, h, oldw, oldh);`
    
  68.      `// 获取单元格的宽和高`
    
  69.      `cellWidth = getMeasuredWidth();`
    
  70.      `int mHeight = getMeasuredHeight();`
    
  71.      `cellHeight = mHeight * 1.0f / LETTERS.length;`
    
  72.  `}`
    
  73. }

代码 解析

  • 代码其实不长,加上一些注释总共才180多行,总体来说,思路分分为以下几个步骤
  1. 在构造方法里面初始化画笔,同时为了使用方便,我们封装了自定义属性

  2. public QuickIndexBar(Context context, AttributeSet attrs, int defStyle) {

  3.      `super(context, attrs, defStyle);`
    
  4.      `mContext = context;`
    
  5.      `//初始化自定义属性`
    
  6.      `obtainAttrs(attrs);`
    
  7.      `// 初始化画笔`
    
  8.      `mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);`
    
  9.      `float textSize = UIUtils.dip2px(15, mContext);`
    
  10.      `mPaint.setTextSize(textSize);`
    
  11.      `mPaint.setTypeface(Typeface.DEFAULT_BOLD);`
    
  12. }

  13.   `private void obtainAttrs(AttributeSet attrs) {`
    
  14.      `TypedArray typedArray = mContext.obtainStyledAttributes(attrs, R.styleable.QuickIndexBar);`
    
  15.      `int selectColor = typedArray.getColor(R.styleable.QuickIndexBar_select_color, -1);`
    
  16.      `if (selectColor != -1) {`
    
  17.          `mSelectColor = selectColor;`
    
  18.      `}`
    
  19.      `int normalColor = typedArray.getColor(R.styleable.QuickIndexBar_normal_color, -1);`
    
  20.      `if (normalColor != -1) {`
    
  21.          `mNormalColor = normalColor;`
    
  22.      `}`
    
  23.      `typedArray.recycle();`
    
  24.  `}`
    
  25. 接着我们在onSizeChange 方法里面拿到我们时间的宽度和高度,有人可能会问了为什么不在onMeasure里面获取了,其实在onMeasure方法里面获取是可以的,只不过我们还需要调用一下measure方法而已,在onSizeChnage方法里面,我们直接调用

  26. cellWidth = getMeasuredWidth();

即可获取到我们需要的宽度和高度。用起来比较方便,不过更多的是为了让大家知道View有这一个方法存在以及怎么使用它

顺便我们来看一下google官方对onSizeChange方法的解释

This is called during layout when the size of this view has changed. If you were just added to the view hierarchy, you’re called with the old values of 0.

从官方的解释我们可以知道这个方法是在onLayout方法中当大小改变的时候会调用这个方法,因此我们直接调用getMeasuredWidth();是可以获取得到宽度的,因为onMeasure 是先于onLayout方法调用的。

3. 接着我们重写onDraw方法,在onDraw方法我们所做的工作就是绘制 我们需要的26个字母

  1.  `protected void onDraw(Canvas canvas) {`
    
  2.      `for (int i = 0; i < LETTERS.length; i++) {`
    
  3.          `String text = LETTERS[i];`
    
  4.          `// 计算坐标`
    
  5.          `int x = (int) (cellWidth / 2.0f - mPaint.measureText(text) / 2.0f);`
    
  6.          `// 获取文本的高度`
    
  7.          `Rect bounds = new Rect();// 矩形`
    
  8.          `mPaint.getTextBounds(text, 0, text.length(), bounds);`
    
  9.          `int textHeight = bounds.height();`
    
  10.          `int y = (int) (cellHeight / 2.0f + textHeight / 2.0f + i * cellHeight);`
    
  11.          `// 根据按下的字母, 设置画笔颜色`
    
  12.          `mPaint.setColor(mLastTouchIndex == i ? mSelectColor : mNormalColor);`
    
  13.          `// 绘制文本A-Z`
    
  14.          `canvas.drawText(text, x, y, mPaint);`
    
  15.      `}`
    
  16.  `}`
    
  17. 讲到这里,我们的工作已经完成一大半了,接着就是处理我们是按下或者一个字母了,我们重写onTouchEvent方法,并且return true;是为了保证 Action_down动作按下以后,Action_move以后的动作能够顺利接受到,这涉及到View的事件分发机制,有空的话我会尝试总结一下,这里就不说了

  18. // 获取当前触摸到的字母索引

  19. index = (int) (event.getY() / cellHeight);

同时我们记录下我们当前是触摸或者按下哪一个字母

  1. mLastTouchIndex = index;

  2. 知道了我们当前是触摸或者按下哪一个字母了,那我们要怎样将这些信息暴露出去了,不难想象就是采用接口回调的方法,为此我们提供了这样一个接口

  3. /**

  4. * 暴露一个字母的监听

  5. */

  6. public interface OnLetterUpdateListener {

  7.  `void onLetterUpdate(String letter);`
    
  8. }

并且提供了设置监听器的方法,这样我们就成功将我们的按下字母的信息提供给外界了

  1. public void setListener(OnLetterUpdateListener listener) {

  2.      `this.listener = listener;`
    
  3. }

详细代码如下

  1. int index = -1;

  2. switch (MotionEventCompat.getActionMasked(event)) {

  3.  `case MotionEvent.ACTION_DOWN:`
    
  4.      `// 获取当前触摸到的字母索引`
    
  5.      `index = (int) (event.getY() / cellHeight);`
    
  6.      `if (index >= 0 && index < LETTERS.length) {`
    
  7.          `// 判断是否跟上一次触摸到的一样,不一样才进行回调`
    
  8.          `if (index != mLastTouchIndex) {`
    
  9.              `if (listener != null) {`
    
  10.                  `//`
    
  11.                  `listener.onLetterUpdate(LETTERS[index]);`
    
  12.              `}`
    
  13.              `Log.d(TAG, "onTouchEvent: " + LETTERS[index]);`
    
  14.              `//记录上一次触摸的Index为当前的index;`
    
  15.              `mLastTouchIndex = index;`
    
  16.          `}`
    
  17.      `}`
    
  18.      `break;`
    
  19.  `case MotionEvent.ACTION_MOVE:`
    
  20.      `index = (int) (event.getY() / cellHeight);`
    
  21.      `if (index >= 0 && index < LETTERS.length) {`
    
  22.          `// 判断是否跟上一次触摸到的一样`
    
  23.          `if (index != mLastTouchIndex) {`
    
  24.              `if (listener != null) {`
    
  25.                  `listener.onLetterUpdate(LETTERS[index]);`
    
  26.              `}`
    
  27.              `Log.d(TAG, "onTouchEvent: " + LETTERS[index]);`
    
  28.              `mLastTouchIndex = index;`
    
  29.          `}`
    
  30.      `}`
    
  31.      `break;`
    
  32.  `case MotionEvent.ACTION_UP:`
    
  33.      `// 手指抬起的时候重置`
    
  34.      `mLastTouchIndex = -1;`
    
  35.      `break;`
    
  36.  `default:`
    
  37.      `break;`
    
  38. }

  39. //调用这个方法会重新调用draw方法,重新绘制

  40. invalidate();

  41. return true;

到此 QuickBarView的源码分析为止,下面我们来学习一下是怎样结合ListView使用的


下面我贴出核心代码,想仔细了解的请点击源码下载 源码下载地址:

  1. mQuickIndexBar.setListener(new OnLetterUpdateListener() {

  2.  `@Override`
    
  3.  `public void onLetterUpdate(String letter) {`
    
  4.      `//          UIUtils.showToast(getApplicationContext(), letter);`
    
  5.      `showLetter(letter);`
    
  6.      `// 根据字母定位ListView, 找到集合中第一个以letter为拼音首字母的对象,得到索引`
    
  7.      `for (int i = 0; i < persons.size(); i++) {`
    
  8.          `Person person = persons.get(i);`
    
  9.          `String l = person.getPinyin().charAt(0) + "";`
    
  10.          `if (TextUtils.equals(letter, l)) {`
    
  11.              `// 匹配成功`
    
  12.              `mListView.setSelection(i);`
    
  13.              `break;`
    
  14.          `}`
    
  15.      `}`
    
  16.  `}`
    
  17. });

思路解析 如下

1. 在我们的List数据里面查找是否有有相应的首字母是触摸的字母,有的话返回相应的index,

2. 然后再调用ListView的setSelection(i)方法选中哪一个Item

  1. mListView.setSelection(i);

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

[外链图片转存中…(img-c0a541DO-1715902154683)]

[外链图片转存中…(img-IqqAwHuY-1715902154686)]

[外链图片转存中…(img-M8Vdv6c2-1715902154686)]

[外链图片转存中…(img-0pGlYYOW-1715902154687)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 21
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值