Android 自定义View 字母索引条

写在开头

这是自定义View的第三篇文章,第一篇是Android drawPath实现QQ拖拽泡泡,主要实现的是题目说的东西,第二篇是Android 自定义View 跳动的水果和文字,可能看这个题目不知道说的是撒,主要讲的是Android drawTextOnPath()的相关方法,以及属性动画相关的使用。当然个人觉得动画效果还是阔以的 嘻嘻。。
这篇主要还是说说在onDraw()drawText()相关的使用,实现的效果就是如图所示!

index_静态.png
index.gif

一个View从出生到你能看到的话,肯定是会经历onMeasure()onLayout()onDraw(),这几方法的,而自定义View无外乎也要涉及到这几个相关的方法,这篇文章没有那么复杂,主要涉及的就是onDraw()方法!

开门见山-IndexBar

不管是在QQ上,还是在163的邮箱中,或者自己手机的通讯录中,右侧都会躺着一个这个玩意儿,我姑且不造官方有没有相关的东西,或者大家约定俗成的称呼这个玩意儿叫什么,反正我就叫它索引条-IndexBar了吧!

IndexBar从整体样式上(我观察的哈),分为两种,一种就是不管三七二十一,26个字母糊糊的贴上去的那种,还有一种就是根据当前的具体内容,只展示相关的首字母的!至于touch到IndexBar背景变为灰色,滑动时选中的字母呈现出选中的状态,这些都搜easy滴!!当然你可能要说还有开头是#号的,或者写着热门等等等的。。

实现思路

这个问题要一分为二来看,首先是怎么把26个字母画出来,然后才是怎么去识别触摸对应的是哪个字母!!

画出对应的字母

这个不用多说,肯定是要调用 drawText()相关的方法,drawText(@NonNull String text, float x, float y, @NonNull Paint paint),这里我们需要注意的就是这里的x和y是撒意思了!
它就是控制这个文字开始的左下定位的坐标。文字就是从这个点的开始向右上绘制出来的!Demo中onDraw方法有对应的注释了的方法,打开可以直接看相关的效果。

首先确定X轴的距离,就是(总的宽度-文字的宽度)/2,这样每个文字水平就是居中显示的了!!
然后确定Y轴的位置,就是(每个文字的总高度+文字的高度)/2,(文字是确定的左下方的坐标点,向下应该加起来!)这样每个文字在竖直方向单位高度中也是居中显示的了!!

那么问题来了,上面说的那些宽度高度等要怎么获取呢?
获取屏幕的高度,平分到26个字母,有'# '或者 ‘热门’再把相关的东西加上!这个就是 每个文字的总高度!接下来就是涉及到这几个方法:
onDraw() 这个不用说,你不draw怎么能展示出来呢?
onSizeChanged(),如果屏幕尺寸发生了变化,不如说虚拟按键隐藏或者展示之后,还有就是屏幕旋转相关的。。
setLetters(),准备好了相关的字母之后,这里就需要去再去计算新的相关参数然后通知绘制。

    @Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    if (letters != null) {
        for (int i = 0; i < letters.size(); i++) {
            String text = letters.get(i);
            float textWidth = mPaint.measureText(text);
            mPaint.getTextBounds(text, 0, text.length(), mRect);
            float textHeight = mRect.height();
            float x = mCellWidth * 0.5f - textWidth * 0.5f;
            float y = mCellHeight * 0.5f + textHeight * 0.5f + mCellHeight * i + beginY;
            mPaint.setColor(mIndex == i ? selecColor : normalColor);
            canvas.drawText(text, x, y, mPaint);

        }
    }
}

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    mHeight = getMeasuredHeight()-getPaddingTop()-getPaddingBottom();
    mCellWidth = getMeasuredWidth();
    mCellHeight = mHeight * 1.0f / 26;
    if (letters != null) {
        beginY = (mHeight - mCellHeight * letters.size()) * 0.5f;
    }

}


public void setLetters(@Nullable List<String> letters) {

    if (letters == null) {
        setVisibility(GONE);
        return;
    }
    this.letters = letters;
    mHeight = getMeasuredHeight()-getPaddingTop()-getPaddingBottom();
    mCellWidth = getMeasuredWidth();
    mCellHeight = mHeight * 1.0f / 26;
    beginY = (mHeight - mCellHeight * letters.size()) * 0.5f;
    invalidate();
}

setLetters()onSizeChanged()里面的代码基本上是重复的,只是在setLetters()里面调用了 invalidate()去通知重新绘制。

触摸的相关状态添加

首先是触摸到这个索引条,背景加深,这个肯定就是走touch事件了嘛,在ACTION_DOWN的时候修改相关状态,在ACTION_UP的时候,再次刷新相关状态咯。

这里要使用refreshDrawableState()onCreateDrawableState()这两个方法,如果你知道了,就当我在这里瞎比比吧!哈哈。。如果不清楚,可以看看我之前写的一篇自定义状态选择器

//定义一个状态
private static final int[] STATE_FOCUSED = new int[]{android.R.attr.state_focused};
//DOWN 设为true UP 设为false
private void refreshState(boolean state) {
    if (pressed != state) {
        pressed = state;
        refreshDrawableState();
    }
}

@Override
public int[] onCreateDrawableState(int extraSpace) {
    int[] states = super.onCreateDrawableState(extraSpace + 1);
    if (pressed) {
        mergeDrawableStates(states, STATE_FOCUSED);
    }
    return states;
}

背景选择基本欧克了!

然后是选中的字母的颜色,这个其实就是更换画笔的颜色就好了!!这个就放在下面的一块内容中。

点击相关回调

用户看到的都是表象,触摸到的肯定是某一个坐标值,这个坐标应该对应这26个字母中的某一个字母的所在的坐标!比如说总高度是2600,然后每个字母Y轴所占的区域就是100,你触摸的坐标是(x,520),那这个明显就是第六个字母了嘛,获取到了对应的position,这个问题就解决完了!

       @Override
public boolean onTouchEvent(MotionEvent event) {
    float y;
    invalidate();
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            Log.i("TAG", "onTouchEvent:Down ");
            getParent().requestDisallowInterceptTouchEvent(true);
            refreshState(true);
            y = event.getY();
            checkIndex(y);
            break;
        case MotionEvent.ACTION_MOVE:
            y = event.getY();
            checkIndex(y);
            break;
        case MotionEvent.ACTION_UP:
            refreshState(false);
            mIndex = -1;
            break;

        default:
            break;
    }
    return true;
}



private void checkIndex(float y) {
    int currentIndex;
    if (y < beginY+getPaddingTop()) {
        return;
    }
    currentIndex = (int) ((y - beginY-getPaddingTop()) / mCellHeight);
    if (currentIndex != mIndex) {
        if (mOnLetterChangeListener != null) {
            if (letters != null && currentIndex < letters.size()) {
                mIndex = currentIndex;
                mOnLetterChangeListener.onLetterChange(letters.get(currentIndex));
    //          Log.i(TAG, "checkIndex: "+letters.get(currentIndex));
            }
        }
       
    }
}

然后就是上面遗留的那个问题,选中字母颜色的更改就是通过这个mIndex来实现的,在draw方法中的这行代码:

mPaint.setColor(mIndex == i ? selecColor : normalColor);
            canvas.drawText(text, x, y, mPaint);

那到这里可以看到,那就是View的展现和相关逻辑的确是分开的,你看到的都是一些表面现象。

滚动到指定的位置

这个是最终的要求了,这里要区分实现机制了,如果你是使用了ListView,那么直接调用setSelection()就可以滚动到指定的位置了。
如果你是使用了RecycleView的话,那么就是使用LayoutManager的manager.scrollToPositionWithOffset(pos,0)
我在测验中发现直接使用manager.scrollToPosition()的话,的确可以滚动,但是不是出现在顶部位置!

总结

本次Indexbar的话,绘制部分主要涉及到了onDraw()方法,canvas.drawText()
细节的话,就是onSizeChanged() 和 setLetters()之后的通知重新绘制。
还有就是状态选择器,两个方法refreshDrawableState()onCreateDrawableState()
需要注意的是,有两个偏移量 beginY 和 topPadding,beginY是用居中的一个偏移量,topPadding就不用多说了!

回调部分,就是onTouch相关处理,根据getY()获取相关Y轴的值推算出对应的position,然后再回调到对应的ListView或者RecycleView

Gif所示的Demo地址:IndexDemo

自定义View的Demo相关地址自定义View

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,560评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,104评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,297评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,869评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,275评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,563评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,833评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,543评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,245评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,512评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,011评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,359评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,006评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,062评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,825评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,590评论 2 273
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,501评论 2 268

推荐阅读更多精彩内容