通讯录A-Z指示器Demo

:)

**from** bing 17/**04**/2**9**

本文主要是介绍了一个自定义控件的实现

  • 首先来看demo效果
**Demo**就是这个样儿 **^_^**
  • 这里有几个问题需要注意
  • 控件的 高度 怎么确定
  • 字体问题每个字符的 字体 都不一样
  • 字体 颜色 问题,当前选中的字符颜色是不一样的
  • 怎么判断当前是 选中 的那个字符

选择实现方案,是继承View还是ViewGroup

  • 继承ViewGroup(比如:LinearLayout),好处就是比较方便实现,不是太好的就是效率不高(要实例化多个子控件)
  • 继承View,好处就是效率较高,自己比较好控制,不爽的就是比较麻烦。

这个demo就选择继承View吧

  • 接下来一个一个解决上面提到的问题(字体,颜色,高度,滑动的位置判断)

总体思路

  • 首先要明白的一点是滑动的监听肯定是通过重写onTouchEvent方法来实现的。这里就还有一个问题,onTouchEvent方法我们只能得到坐标信息怎么通过坐标信息转变成当前是滑动到那个字符?demo采用的解决方法是将每个字符对应的坐标都保存到一个实体里面,判断位置的时候就根据每个字符的坐标信息就可以判断是滑动到那个实体。

  • 字体大小怎么确定?demo里面规定了一个最大的字体最小的字体,当前选中的字符使用最大的字体,其他字符就根据它离最大字体的位置来设置字体大小

  • 控件的高度的确定?其实就是当选中中间的字符的时候的高度


下面有大量代码

                                [↓][↓][↓]
                                [↓][↓][↓]
                                [↓][↓][↓]
                          [↓][↓][↓][↓][↓][↓][↓]
                                [↓][↓][↓]
                                   [↓]

定义控件会用到的属性

/**
     *
     * 最大的字体
     *
     */
    private int maxTextSize = DEFAULT_MAX_TEXT_SIZE;

    /**
     *
     * 最小的字体
     *
     */
    private int minTextSize = DEFAULT_MIN_TEXT_SIZE;

    /**
     *
     * 不同字体之间的间隔
     *
     */
    private int textSizeStep = DEFAULT_TEXT_STEP;

    /**
     *
     * 选中的时候字体颜色
     *
     */
    private int choiseTextColor = DEFAULT_CHOISE_COLOR;

    /**
     *
     * 默认的字体颜色
     *
     */
    private int defaultTextColor = DEFAULT_COLOR;


    /**
     *
     * 字符上下间距
     *
     *
     */
    private int margin = 8;

    private static final int DEFAULT_MAX_TEXT_SIZE = 100;//px
    private static final int DEFAULT_MIN_TEXT_SIZE = 15;
    private static final int DEFAULT_TEXT_STEP = 10;
    private static final int DEFAULT_CHOISE_COLOR = Color.BLACK;
    private static final int DEFAULT_COLOR = Color.GRAY;

    /**
     *
     * 默认的选中位置
     *
     *
     */
    private int currentChoise = 0;

建立一个实体类来保存每个字符的位置信息(在滑动和绘制的时候判断位置)

class TextSizeModel {

        private int x;
        private int y;
        private int textSize;

        private String item;
    }
  • 会为每个字符都生成一个这样的实体,实体里面的textSize保存当前字符的字体大小,当选中位置改变之后就可以更新这个实体类,最后在绘制的时候根据这个实体类的信息来进行绘制就可以了

控件高度的确定(这个肯定就是重写 onMeasure了)


 @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

        int width = maxTextSize + margin * 2;
        int height = margin * 2;

        if(datas == null || datas.size() == 0){
            setMeasuredDimension(width,height);
            return;
        }

        currentChoise = datas.size() / 2;
        //height += initItemTextSize();

        for(int i = 0;i < datas.size();i++){
            int duration = i < currentChoise ? currentChoise - i : i - currentChoise;
            int textSize = maxTextSize - duration * textSizeStep;
            textSize = textSize < minTextSize ? minTextSize : textSize;
            datas.get(i).setTextSize(textSize);

            height += textSize + margin;
        }

        setMeasuredDimension(width,height);

    }

  • 这里的datas就是所有字符的实体类

  • currentChoise = datas.size() / 2 这个主要是当选中中间位置的时候控件的高度是最高的

  • 高度的计算方法其实也比较简单,就是看当前字符距离当前选中字符的个数,textSize = 最大的字体 - 当前的距离 * 字体偏移量

  • 这里还有一个要注意的是 onMeasure 里面初始化了字体的大小


有了字体大小,接下来就是绘制了(onDraw)

@Override
    protected void onDraw(Canvas canvas) {

        int currentY = margin + maxTextSize;
        for(int i = 0;i < datas.size();i++){
            paint.setColor(i == currentChoise ? choiseTextColor : defaultTextColor);
            paint.setTextSize(datas.get(i).getTextSize());
            canvas.drawText(datas.get(i).getItem(),
                    getMeasuredWidth() / 2,currentY,paint);
            datas.get(i).setX(0);
            datas.get(i).setY(currentY);
            currentY += datas.get(i).getTextSize() + margin;
        }

    }

  • 这里其实就是便利实体类集合,根据onMesure里面测量的字体大小就行字符的绘制就可以了
  • 还有一个需要注意的是这里进行的实体的坐标的设置

滑动的监听(onTouchEvent)


@Override
    public boolean onTouchEvent(MotionEvent event) {
        int x = (int) event.getX();
        int y = (int) event.getY();

        switch (event.getAction()){
            case MotionEvent.ACTION_DOWN:
                break;
            case MotionEvent.ACTION_MOVE:
                for(int i=0;i < datas.size();i++){
                    TextSizeModel model = datas.get(i);
                    if(model.getY() < y && model.getY() + margin > y){
                        currentChoise = i;
                        break;
                    }
                }
                initItemTextSize();
                postInvalidate();
                break;
            case MotionEvent.ACTION_UP:
                break;
        }

        return true;
    }

  • 主要就是在ACTION_MOVE的时候进行坐标判断,根据上一次绘制的时候实体的坐标和当前坐标更新选中的位置,最后initItemTextSize重新根据当前的选中位置更新了每个字符的字体大小,之后绘制

源码

package com.suse.yuxin.emptydemo.view;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;

import com.suse.yuxin.emptydemo.R;

import java.util.ArrayList;
import java.util.List;

/**
 * Created by Administrator on 2017/4/29.
 */

public class LettersView extends View{

    /**
     *
     * 最大的字体
     *
     */
    private int maxTextSize = DEFAULT_MAX_TEXT_SIZE;

    /**
     *
     * 最小的字体
     *
     */
    private int minTextSize = DEFAULT_MIN_TEXT_SIZE;

    /**
     *
     * 不同字体之间的间隔
     *
     */
    private int textSizeStep = DEFAULT_TEXT_STEP;

    /**
     *
     * 选中的时候字体颜色
     *
     */
    private int choiseTextColor = DEFAULT_CHOISE_COLOR;

    /**
     *
     * 默认的字体颜色
     *
     */
    private int defaultTextColor = DEFAULT_COLOR;


    /**
     *
     * 字符上下间距
     *
     *
     */
    private int margin = 8;

    private static final int DEFAULT_MAX_TEXT_SIZE = 100;//px
    private static final int DEFAULT_MIN_TEXT_SIZE = 15;
    private static final int DEFAULT_TEXT_STEP = 10;
    private static final int DEFAULT_CHOISE_COLOR = Color.BLACK;
    private static final int DEFAULT_COLOR = Color.GRAY;

    /**
     *
     * 默认的选中位置
     *
     *
     */
    private int currentChoise = 0;

    private Paint paint;
    private List<TextSizeModel> datas = new ArrayList<>();


    private static final String LOG_TAG = "LettersView_TAG";

    public LettersView(Context context) {
        super(context);
        init(context,null);
    }

    public LettersView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        init(context,attrs);
    }

    public LettersView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init(context,attrs);
    }


    private void init(Context context,AttributeSet attrs) {
        if(attrs == null){
            return;
        }
        //init attr
        TypedArray arr = context.obtainStyledAttributes(attrs, R.styleable.LettersView);
        maxTextSize = (int) arr.getDimension(R.styleable.LettersView_maxTextSize,DEFAULT_MAX_TEXT_SIZE);
        minTextSize = (int) arr.getDimension(R.styleable.LettersView_minTextSize,DEFAULT_MIN_TEXT_SIZE);
        textSizeStep = (int) arr.getDimension(R.styleable.LettersView_textStep,DEFAULT_TEXT_STEP);
        choiseTextColor = arr.getColor(R.styleable.LettersView_choiseTextColor,DEFAULT_CHOISE_COLOR);
        defaultTextColor = arr.getColor(R.styleable.LettersView_defaultTextColor,DEFAULT_COLOR);
        arr.recycle();

        Log.i(LOG_TAG,"maxT "+maxTextSize+"  minT"+minTextSize+"   step "+textSizeStep);

        paint = new Paint();
        paint.setAntiAlias(true);
        paint.setTextAlign(Paint.Align.CENTER);
    }


    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

        int width = maxTextSize + margin * 2;
        int height = margin * 2;

        if(datas == null || datas.size() == 0){
            setMeasuredDimension(width,height);
            return;
        }

        currentChoise = datas.size() / 2;
        //height += initItemTextSize();

        for(int i = 0;i < datas.size();i++){
            int duration = i < currentChoise ? currentChoise - i : i - currentChoise;
            int textSize = maxTextSize - duration * textSizeStep;
            textSize = textSize < minTextSize ? minTextSize : textSize;
            datas.get(i).setTextSize(textSize);

            height += textSize + margin;
        }

        setMeasuredDimension(width,height);

    }


    private int initItemTextSize(){
        int height = 0;
        for(int i = 0;i < datas.size();i++){
            int duration = i < currentChoise ? currentChoise - i : i - currentChoise;
            int textSize = maxTextSize - duration * textSizeStep;
            textSize = textSize < minTextSize ? minTextSize : textSize;
            datas.get(i).setTextSize(textSize);

            height += textSize + margin;
        }
        return height;
    }

    @Override
    protected void onDraw(Canvas canvas) {

        int currentY = margin + maxTextSize;
        for(int i = 0;i < datas.size();i++){
            paint.setColor(i == currentChoise ? choiseTextColor : defaultTextColor);
            paint.setTextSize(datas.get(i).getTextSize());
            canvas.drawText(datas.get(i).getItem(),
                    getMeasuredWidth() / 2,currentY,paint);
            datas.get(i).setX(0);
            datas.get(i).setY(currentY);
            currentY += datas.get(i).getTextSize() + margin;
        }

    }


    @Override
    public boolean onTouchEvent(MotionEvent event) {
        int x = (int) event.getX();
        int y = (int) event.getY();

        switch (event.getAction()){
            case MotionEvent.ACTION_DOWN:
                break;
            case MotionEvent.ACTION_MOVE:
                for(int i=0;i < datas.size();i++){
                    TextSizeModel model = datas.get(i);
                    if(model.getY() < y && model.getY() + margin > y){
                        currentChoise = i;
                        break;
                    }
                }
                initItemTextSize();
                postInvalidate();
                break;
            case MotionEvent.ACTION_UP:
                break;
        }

        return true;
    }



    public void setData(List<String> letters){
        if(letters == null || letters.size() == 0){
            return;
        }
        datas.clear();
        for (String letter : letters) {
            TextSizeModel sizeModel = new TextSizeModel();
            sizeModel.setItem(letter);
            datas.add(sizeModel);
        }
        requestLayout();
    }


    public void setData(String[] letters){
        if(letters == null || letters.length == 0){
            return;
        }
        List<String> l = new ArrayList<>();
        for(int i=0;i < letters.length;i++){
            l.add(letters[i]);
        }
        setData(l);
    }


    class TextSizeModel {

        private int x;
        private int y;
        private int textSize;

        private String item;

        public int getX() {
            return x;
        }

        public void setX(int x) {
            this.x = x;
        }

        public int getY() {
            return y;
        }

        public void setY(int y) {
            this.y = y;
        }

        public int getTextSize() {
            return textSize;
        }

        public void setTextSize(int textSize) {
            this.textSize = textSize;
        }

        public String getItem() {
            return item;
        }

        public void setItem(String item) {
            this.item = item;
        }
    }

}

<declare-styleable name="LettersView">
        <attr name="maxTextSize" format="dimension"/>
        <attr name="minTextSize" format="dimension"/>
        <attr name="choiseTextColor" format="color"/>
        <attr name="defaultTextColor" format="color"/>
        <attr name="textStep" format="color"></attr>
    </declare-styleable>


如何使用


public class LettersTestActivity  extends AppCompatActivity{


    LettersView lettersView;


    String[] letters = new String[]{
            "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"
    };

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_letters_test);
        lettersView = (LettersView) findViewById(R.id.activity_letters_test_content);
        lettersView.setData(letters);
    }
}

Nothing is certain in this life. The only thing i know for sure is that. I love you and my life. That is the only thing i know. have a good day

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,830评论 25 707
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,373评论 0 17
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,066评论 4 62
  • 每一次与别人交往的机会都非常重要,如果你能将遇到的对象的相关情况都详细记录下来,就可以拓展自己的人脉。这样晨间日记...
    笑飞飞阅读 179评论 0 3
  • 我想经历了很多,我学到了不少。该柔软的时候就柔软,该大度的时候就大度,想哭就哭,想笑就笑。但同时,该狠心的时候就狠...
    兔子1235阅读 478评论 0 0