Android--快速定位索引

先看张效果图

device-2018-06-20-095811.gif

列表是用RecyclerView,demo中RecyclerView都是封装好的,小编是直接拿过来用的,至于怎么封装RecyclerView,绘制RecyclerView的分割线等等,有个半年经验左右应该是会封装的,至于不会,那就研究下我的代码是怎么封装的。哈哈!

具体思路:

1、对汉字进行A_Z排序
2、绘制字母A-Z索引LetterSideBarView  
3、触摸响应手指的touch事件,需要考虑View的事件分发   
4、绘制RecyclerView的分割线  

LetterSideBarView

public class LetterSideBarView extends View {

private Context mContext;
//画笔
private Paint mPaint;
private String[] mLetters = {"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 String mTouchLetter;
//手指是否触摸
private boolean mCurrentIsTouch;
// 设置触摸监听
private SideBarTouchListener mTouchListener;

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


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

public LetterSideBarView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    this.mContext = context;
    mPaint = new Paint();
    mPaint.setAntiAlias(true);
    mPaint.setTextSize(DensityUtil.sp2px(this.getContext(), 14));
    mPaint.setColor(Color.BLACK);
}

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    //每一个字母的高度
    float singleHeight = ( float ) getHeight() / mLetters.length;
    //不断循环把绘制字母
    for (int i = 0; i < mLetters.length; i++) {
        String letter = mLetters[i];
        //获取字体的宽度
        Rect rect = new Rect();
        mPaint.getTextBounds(letter, 0, letter.length(), rect);
        float measureTextWidth = rect.width();
        //获取内容的宽度
        int contentWidth = getWidth() - getPaddingLeft() - getPaddingRight();
        float x = getPaddingLeft() + (contentWidth - measureTextWidth) / 2;
        //计算基线位置
        Paint.FontMetrics fontMetrics = mPaint.getFontMetrics();
        float baseLine = singleHeight / 2 + (singleHeight * i) +
                (fontMetrics.bottom - fontMetrics.top) / 2 - fontMetrics.bottom;
        //画字母,后面onTouch的时候需要处理高亮
        if (mLetters[i].equals(mTouchLetter) && mCurrentIsTouch) {
            mPaint.setTextSize(DensityUtil.sp2px(mContext, 18));
            mPaint.setColor(Color.RED);
            canvas.drawText(letter, x, baseLine, mPaint);
        } else {
            mPaint.setTextSize(DensityUtil.sp2px(mContext, 14));
            mPaint.setColor(Color.BLACK);
            canvas.drawText(letter, x, baseLine, mPaint);
        }
    }
}

@Override
public boolean onTouchEvent(MotionEvent event) {
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
        case MotionEvent.ACTION_MOVE:
            //计算出当前触摸的字母,获取当前的位置
            float currentMoveY = ( int ) event.getY();
            int itemHeight = (getHeight() - getPaddingTop() - getPaddingBottom()) / mLetters.length;
            //当前的位置=currentMoveY/字母的高度
            int currentPosition = ( int ) currentMoveY / itemHeight;
            if (currentPosition < 0) {
                currentPosition = 0;
            }
            if (currentPosition > mLetters.length - 1) {
                currentPosition = mLetters.length - 1;
            }
            mTouchLetter = mLetters[currentPosition];
            mCurrentIsTouch = true;
            if (mTouchListener != null) {
                mTouchListener.onTouch(mTouchLetter, true);
            }
            break;
        case MotionEvent.ACTION_UP:
            mCurrentIsTouch = false;
            if (mTouchListener != null) {
                mTouchListener.onTouch(mTouchLetter, false);
            }
            break;
    }
    invalidate();

    return true;
}


public void setOnSideBarTouchListener(SideBarTouchListener touchListener) {
    this.mTouchListener = touchListener;
}

计算出每个字母的基线baseLine,怎么计算呢?

5437668-8051a93a8a68dd67.png
  • top line: 文字可绘制区域最顶部的线;
  • ascent line: 系统推荐的,文字可绘制区域顶部的线;
  • baseline: 文字绘制的基线(在四线格上书写英文字母时的第三条线);
  • descent line: 系统推荐的,文字可绘制区域底部的线;
  • bottom line: 文字可绘制区域最底部的线。
    baseline是基线,baseline以上是负值,baseline以下是正值,因此ascent和top都是负值,descent和bottom都是正值。
//计算基线位置
 Paint.FontMetrics fontMetrics = mPaint.getFontMetrics();
 float baseLine = singleHeight / 2 + (singleHeight * i) +(fontMetrics.bottom - fontMetrics.top) / 2 - fontMetrics.bottom;

说下View的事件分发,需要执行ACTION_MOVE事件,ACTION_DOWN事件需要返回true消费这个事件,才能往下执行ACTION_MOVE事件。

MainActivity

public class MainActivity extends AppCompatActivity {
private RecyclerView mRv;
private LetterSideBarView mLetterSideBarView;
private TextView mIndexTv;
private List<Person> mList;
private Handler mHandler = new Handler();
private boolean isScale = false;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    mRv = findViewById(R.id.rv);
    mLetterSideBarView = findViewById(R.id.letterSideBarView);
    mIndexTv = findViewById(R.id.indexTv);
    initRv();
    mLetterSideBarView.setOnSideBarTouchListener(new SideBarTouchListener() {
        @Override
        public void onTouch(String letter, boolean isTouch) {
            for (int i = 0; i < mList.size(); i++) {
                if (letter.equals(mList.get(i).getPinyin().charAt(0) + "")) {
                    mRv.scrollToPosition(i);
                    break;
                }
            }
            showCurrentIndex(letter);
        }
    });
}

private void showCurrentIndex(String letter) {
    mIndexTv.setText(letter);
    if (!isScale) {
        isScale = true;
        ViewCompat.animate(mIndexTv)
                .scaleX(1f)
                .setInterpolator(new OvershootInterpolator())
                .setDuration(380)
                .start();
        ViewCompat.animate(mIndexTv)
                .scaleY(1f)
                .setInterpolator(new OvershootInterpolator())
                .setDuration(380)
                .start();
    }

    mHandler.removeCallbacksAndMessages(null);
    // 延时隐藏
    mHandler.postDelayed(new Runnable() {
        @Override
        public void run() {
            ViewCompat.animate(mIndexTv)
                    .scaleX(0f)
                    .setDuration(380)
                    .start();
            ViewCompat.animate(mIndexTv)
                    .scaleY(0f)
                    .setDuration(380)
                    .start();
            isScale = false;
        }
    }, 380);
}

private void initRv() {
    mList = new ArrayList<>();
    for (int i = 0; i < DataUtil.testData3.length; i++) {
        Person person = new Person(DataUtil.testData3[i]);
        mList.add(person);
    }
    //排序
    Collections.sort(mList);
    PersonAdapter adapter = new PersonAdapter(this, mList, R.layout.person_recycler_item);
    mRv.setAdapter(adapter);
    mRv.setLayoutManager(new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false)); 
    //添加分割线
    mRv.addItemDecoration(new DefaultItemDecoration(this, R.drawable.default_item));
}

private class PersonAdapter extends CommonRecycleAdapter<Person> {
    public PersonAdapter(Context context, List<Person> mData, int layoutId) {
        super(context, mData, layoutId);
    }

    @Override
    protected void convert(CommonViewHolder holder, Person person, int position) {
        String currentWord = person.getPinyin().charAt(0) + "";
        if (position > 0) {
            String lastWord = mList.get(position - 1).getPinyin().charAt(0) + "";
            //拿当前的首字母和上一个首字母比较,与首字母相同,需要隐藏当前item的索引
            holder.setVisibility(R.id.indexTv, currentWord.equals(lastWord) ? View.GONE : View.VISIBLE);
        } else {
            holder.setVisibility(R.id.indexTv, View.VISIBLE);
        }
        holder.setText(R.id.indexTv, currentWord);
        holder.setText(R.id.userNameTv, person.getName());
    }
}
}

MainActivity大家看代码吧,都是些基础代码。主要是这个:拿当前的首字母和上一个首字母比较,与首字母相同,需要隐藏当前item上面索引。

github地址:https://github.com/StevenYan88/QuickIndex

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

推荐阅读更多精彩内容

  • 我奶奶今年九十了。 两只耳朵重度耳聋,要凑近了喊,才能听到。 膝盖有骨刺,不能走太多路。 一辈子活在上海的郊区,听...
    夏天的威尼阅读 329评论 0 0