Android 側(cè)邊觸摸式導(dǎo)航欄

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

先上動(dòng)態(tài)效果圖


image.png

利用一個(gè)自定義View 虫蝶,和其中的dispatchTouchEvent 攔截觸摸事件實(shí)現(xiàn)

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();//獲取每一個(gè)字母所占的高度
        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軸坐標(biāo)
            float xPos=width/2-paint.measureText(letterList.get(i))/2;
 
            //文字y軸坐標(biāo)
//            float yPos=(singleHeight*(i)) + (2*singleHeight/3);
            float yPos=(singleHeight*i)+((singleHeight+textHight)/2);
 
            canvas.drawText(letterList.get(i),xPos,yPos,paint);
            paint.reset();//重置畫(huà)筆
        }
    }
 
 
    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        int action=event.getAction();
        final float x=event.getX();
        final float y=event.getY();
 
        // 點(diǎn)擊y坐標(biāo)所占總高度的比例*b數(shù)組的長(zhǎng)度就等于點(diǎn)擊b中的個(gè)數(shù).
        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;//要實(shí)現(xiàn)觸摸式的動(dòng)態(tài)效果,必須返回true裕寨,使得事件能夠被攔截
 
 
    }
 
    //獲得字體baseLine以上的高度
    //若需要了解關(guān)于單行文本高度分析詳情請(qǐng)看 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));
    }
 
 
    //外部設(shè)置列表數(shù)據(jù)方法攒读,重繪view
    public void setLetterList(List<String> list){
        this.letterList=list;
        invalidate();
    }
 
 
 
    //外部調(diào)用接口廓旬,以便讓外部獲得當(dāng)前觸摸到的字母
    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);
 
        //屏幕中間顯示的文本凿蒜,會(huì)將當(dāng)前觸摸到的字母顯示出來(lái)
        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);
 
 
        //包裹側(cè)邊導(dǎo)航欄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);
 
 
        //包裹側(cè)邊導(dǎo)航欄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內(nèi)容 重繪view
        Button button =new Button(getBaseContext());
        rp=new RelativeLayout.LayoutParams(WC,WC);
        button.setText("點(diǎn)我");
        rlayout.addView(button,rp);
 
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
 
                letterList= Arrays.asList(Afterstring);
                sideBar.setLetterList(letterList);
 
            }
        });
 
    }
}

若需要了解關(guān)于單行文本高度分析詳情請(qǐng)看 http://www.reibang.com/p/aab850e29b6c

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

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市绝编,隨后出現(xiàn)的幾起案子僻澎,更是在濱河造成了極大的恐慌,老刑警劉巖瓮增,帶你破解...
    沈念sama閱讀 211,194評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件怎棱,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡绷跑,警方通過(guò)查閱死者的電腦和手機(jī)拳恋,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)砸捏,“玉大人谬运,你說(shuō)我怎么就攤上這事】巡兀” “怎么了梆暖?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,780評(píng)論 0 346
  • 文/不壞的土叔 我叫張陵,是天一觀(guān)的道長(zhǎng)掂骏。 經(jīng)常有香客問(wèn)我轰驳,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,388評(píng)論 1 283
  • 正文 為了忘掉前任级解,我火速辦了婚禮冒黑,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘勤哗。我一直安慰自己抡爹,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布芒划。 她就那樣靜靜地躺著冬竟,像睡著了一般。 火紅的嫁衣襯著肌膚如雪民逼。 梳的紋絲不亂的頭發(fā)上泵殴,一...
    開(kāi)封第一講書(shū)人閱讀 49,764評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音拼苍,去河邊找鬼袋狞。 笑死,一個(gè)胖子當(dāng)著我的面吹牛映屋,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播同蜻,決...
    沈念sama閱讀 38,907評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼棚点,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了湾蔓?” 一聲冷哼從身側(cè)響起瘫析,我...
    開(kāi)封第一講書(shū)人閱讀 37,679評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎默责,沒(méi)想到半個(gè)月后贬循,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,122評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡桃序,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評(píng)論 2 325
  • 正文 我和宋清朗相戀三年杖虾,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片媒熊。...
    茶點(diǎn)故事閱讀 38,605評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡奇适,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出芦鳍,到底是詐尸還是另有隱情嚷往,我是刑警寧澤,帶...
    沈念sama閱讀 34,270評(píng)論 4 329
  • 正文 年R本政府宣布柠衅,位于F島的核電站皮仁,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜贷祈,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評(píng)論 3 312
  • 文/蒙蒙 一趋急、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧付燥,春花似錦宣谈、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,734評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至勋颖,卻和暖如春嗦嗡,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背饭玲。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,961評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工侥祭, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人茄厘。 一個(gè)月前我還...
    沈念sama閱讀 46,297評(píng)論 2 360
  • 正文 我出身青樓矮冬,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親次哈。 傳聞我的和親對(duì)象是個(gè)殘疾皇子胎署,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評(píng)論 2 348