Android 侧边触摸式导航栏

原文:https://blog.csdn.net/uyy203/article/details/54912969

先上动态效果图


image.png

利用一个自定义View ,和其中的dispatchTouchEvent 拦截触摸事件实现

SideBar.java

package xyz.slideviewgettext;
 
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Typeface;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
 
import java.util.List;
 
/**
 * Created by xyz on 2017/2/5.
 */
 
public class SideBar extends View {
    private List<String> letterList;
    private Paint paint;
 
    public SideBar(Context context,List<String> list){
        this(context,(AttributeSet) null);
        this.letterList=list;
    }
 
    public  SideBar(Context context, AttributeSet attributeSet){
        this(context,attributeSet,0);
    }
 
    public  SideBar(Context context,AttributeSet attributeSet,int defStyle){
        super(context,attributeSet,defStyle);
        init();
    }
 
    private void init(){
        setBackgroundColor(0x000000);
    }
 
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        int height=getHeight();//获取view的高度
        int width=getWidth();//获取view的宽度
        int singleHeight=height/letterList.size();//获取每一个字母所占的高度
        int textHight= getFontAboveBaseLineHeight(35);
 
 
        paint=new Paint();
 
 
        for(int i=0;i<letterList.size();i++){
            paint.setColor(0xff606060);
            paint.setTypeface(Typeface.DEFAULT_BOLD);
            paint.setAntiAlias(true);
            paint.setTextSize(35);
 
            //文字x轴坐标
            float xPos=width/2-paint.measureText(letterList.get(i))/2;
 
            //文字y轴坐标
//            float yPos=(singleHeight*(i)) + (2*singleHeight/3);
            float yPos=(singleHeight*i)+((singleHeight+textHight)/2);
 
            canvas.drawText(letterList.get(i),xPos,yPos,paint);
            paint.reset();//重置画笔
        }
    }
 
 
    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        int action=event.getAction();
        final float x=event.getX();
        final float y=event.getY();
 
        // 点击y坐标所占总高度的比例*b数组的长度就等于点击b中的个数.
        final int index=(int)((y/getHeight())*letterList.size());
 
        if(x<0||x>getWidth()||y<0||y>getHeight()) {
            action = MotionEvent.ACTION_UP;
        }
 
            switch (action) {
 
                case MotionEvent.ACTION_UP:
                    setBackgroundColor(0x000000);
                    if (onTouchBarListener != null) {
                        onTouchBarListener.onTouch(null, -1, false);
 
                    }
 
                    invalidate();
                    break;
 
 
                default:
                    setBackgroundColor(0xffffffff);
                    if (onTouchBarListener != null) {
                        onTouchBarListener.onTouch(letterList.get(index), index, true);
                    }
 
 
                        invalidate();
                        break;
 
 
 
            }
 
 
        return true;//要实现触摸式的动态效果,必须返回true,使得事件能够被拦截
 
 
    }
 
    //获得字体baseLine以上的高度
    //若需要了解关于单行文本高度分析详情请看 http://blog.csdn.net/uyy203/article/details/54926753
    public int getFontAboveBaseLineHeight(float fontSize)
    {
        Paint paint = new Paint();
        paint.setTextSize(fontSize);
        Paint.FontMetrics fm = paint.getFontMetrics();
        return (int)(Math.ceil(- fm.ascent));
    }
 
 
    //外部设置列表数据方法,重绘view
    public void setLetterList(List<String> list){
        this.letterList=list;
        invalidate();
    }
 
 
 
    //外部调用接口,以便让外部获得当前触摸到的字母
    private OnTouchBarListener onTouchBarListener;
    public interface OnTouchBarListener{
        void onTouch(String letter,int pos,boolean state);
    }
    public void setOnTouchBarLisstener(OnTouchBarListener l){
        this.onTouchBarListener=l;
    }
 
 
}

MainActivity.java

package xyz.slideviewgettext;
 
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.view.Window;
import android.widget.Button;
import android.widget.RelativeLayout;
import android.widget.TextView;
 
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
 
public class MainActivity extends Activity {
    public List<String> letterList;
 
    public static String[] INDEX_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",
            "#"
    };
 
    public static String[] Afterstring = {
            "1", "2", "3", "4", "5", "6", "7", "8", "9",
            "@"
    };
 
 
    private RelativeLayout rlayout;
    private static int WC= ViewGroup.LayoutParams.WRAP_CONTENT;
    private static int MP=ViewGroup.LayoutParams.MATCH_PARENT;
    private SideBar sideBar;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
//        setContentView(R.layout.activity_main);
 
        letterList=new ArrayList<String>();
        letterList= Arrays.asList(INDEX_STRING);
 
        //整体布局
        rlayout=new RelativeLayout(getBaseContext());
        RelativeLayout.LayoutParams rp=new RelativeLayout.LayoutParams(MP,MP);
        setContentView(rlayout,rp);
 
        //屏幕中间显示的文本,会将当前触摸到的字母显示出来
        final TextView text=new TextView(getBaseContext());
        rp=new RelativeLayout.LayoutParams(WC,WC);
        rp.addRule(RelativeLayout.CENTER_IN_PARENT);
        text.setTextSize(100);
        text.setTextColor(0xff606060);
        rlayout.addView(text,rp);
 
 
        //包裹侧边导航栏view的布局
        final RelativeLayout sideLayout=new RelativeLayout(getBaseContext());
        rp=new RelativeLayout.LayoutParams(80,letterList.size()*60);
        rp.addRule(RelativeLayout.ALIGN_PARENT_RIGHT);
        rp.addRule(RelativeLayout.CENTER_VERTICAL);
        rlayout.addView(sideLayout,rp);
 
 
        //包裹侧边导航栏view
        sideBar=new SideBar(getBaseContext(),letterList);
        rp=new RelativeLayout.LayoutParams(WC,WC);
        sideLayout.addView(sideBar,rp);
 
 
        sideBar.setOnTouchBarLisstener(new SideBar.OnTouchBarListener() {
            @Override
            public void onTouch(String letter, int pos,boolean state) {
                if(state)
                    text.setText(letter);
                else
                    text.setText("");
            }
        });
 
 
        //改变list内容 重绘view
        Button button =new Button(getBaseContext());
        rp=new RelativeLayout.LayoutParams(WC,WC);
        button.setText("点我");
        rlayout.addView(button,rp);
 
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
 
                letterList= Arrays.asList(Afterstring);
                sideBar.setLetterList(letterList);
 
            }
        });
 
    }
}

若需要了解关于单行文本高度分析详情请看 https://www.jianshu.com/p/aab850e29b6c

github: https://github.com/Cedric-Xuan/slideViewGetText

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