Android開(kāi)發(fā)藝術(shù)探索筆記 ——View(一)

Android開(kāi)發(fā)藝術(shù)探索筆記 ——View(一)

View的基礎(chǔ)知識(shí)

什么是View

View是Android中所有控件的基類。是一種界面層控件的抽象

View的位置參數(shù)

Paste_Image.png
Paste_Image.png
參數(shù)名 獲取方式 含義
top getTop() View左上角縱坐標(biāo)
left getLeft() View左上角橫坐標(biāo)
bottom getBottom() View右下角縱坐標(biāo)
right getRight() View右下角橫坐標(biāo)
View:x/y getX()/getY() View左上角坐標(biāo)
translationX getTranslationX() View左上角相對(duì)于父容器的偏移量
translationY getTranslationY() View左上角相對(duì)于父容器的偏移量
MotionEvent:x/y event.getX()/event.getY() 點(diǎn)擊事件相對(duì)于View左上角的坐標(biāo)
MotionEvent:rawX/rawY event.getRawX()/event.getRawY() 點(diǎn)擊事件相對(duì)于手機(jī)屏幕左上角的坐標(biāo)

MotionEvent

  • ACTION_DOWN
  • ACTION_MOVE
  • ACTION_UP

MotionEvent獲取點(diǎn)擊事件x,y坐標(biāo):

getX/getY 相對(duì)于View左上角
getRawX/getRawY 相對(duì)于手機(jī)屏幕左上角

  //在View或Activity中攔截touch events顶瞒,重寫onTouchEvent方法
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        int action = MotionEventCompat.getActionMasked(event);
        switch (action) {
            case (MotionEvent.ACTION_DOWN):
                Log.d(TAG,"action down");
                return true;
            case (MotionEvent.ACTION_MOVE):
                Log.d(TAG,"action move");
                return true;
            case (MotionEvent.ACTION_UP):
                Log.d(TAG,"action up");
                return true;
            case (MotionEvent.ACTION_CANCEL):
                Log.d(TAG,"action cancel");
                return true;
            case (MotionEvent.ACTION_OUTSIDE):
                Log.d(TAG,"action outside");
                return true;
            default:
                return super.onTouchEvent(event);
        }
    }
    //對(duì)于View,可以使用setOnTouchListener()方法來(lái)監(jiān)聽(tīng)touch events而不需要繼承現(xiàn)有的View
    View myView = findViewById(R.id.my_view);
myView.setOnTouchListener(new OnTouchListener() {
    public boolean onTouch(View v, MotionEvent event) {
        // ... Respond to touch events
        return true;
    }
});

TouchSlop

滑動(dòng)的最小距離,常量留搔,和設(shè)備有關(guān)。

//獲取方式:
ViewConfiguration.get(getContext()).getScaledTouchSlop();

VelocityTracker

學(xué)習(xí)資料:
1.https://developer.android.com/training/gestures/movement.html

追蹤手指滑動(dòng)的速度

使用過(guò)程:

  //在View的onTouchEvent方法中
  VelocityTracker velocityTracker = VelocityTracker.obtain();
  velocityTracker.addMoveMent(event);
  
  //時(shí)間間隔設(shè)為1000ms
  tracker.computeCurrentVelocity(1000);
  int Vx = (int)velocityTracker.getXVelocity();
  int Vy = (int)VelocityTracker.getYVelocity();
  
  //不再使用的時(shí)候铛铁,需要清除并回收
  velocityTracker.clear();
  velocityTracker.recycle();

GestureDetector

手勢(shì)監(jiān)測(cè):單擊隔显,滑動(dòng),長(zhǎng)按饵逐,雙擊等等括眠。

使用方法:

  • 1.實(shí)例化GestureDetectorCompat
  • 2.如需監(jiān)測(cè)所有的手勢(shì),則實(shí)現(xiàn)GestureDetector.OnGestureListener接口和GestureDetector.OnDoubleTapListener接口倍权。
  • 3.若只需監(jiān)測(cè)部分手勢(shì)掷豺,則繼承GestureDetector.SimpleOnGestureListener類。

//監(jiān)測(cè)所有的手勢(shì)
public class MainActivity extends Activity implements
        GestureDetector.OnGestureListener,
        GestureDetector.OnDoubleTapListener{

    private static final String DEBUG_TAG = "Gestures";
    private GestureDetectorCompat mDetector;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //實(shí)例化GestureDetectorCompat
        mDetector = new GestureDetectorCompat(this,this);
        //將雙擊事件交給GestureDetectorCompat監(jiān)聽(tīng)
        mDetector.setOnDoubleTapListener(this);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event){
        this.mDetector.onTouchEvent(event);
        return super.onTouchEvent(event);
    }

    @Override
    public boolean onDown(MotionEvent event) {
        Log.d(DEBUG_TAG,"onDown: " + event.toString());
        return true;
    }

    @Override
    public boolean onFling(MotionEvent event1, MotionEvent event2,
            float velocityX, float velocityY) {
        Log.d(DEBUG_TAG, "onFling: " + event1.toString()+event2.toString());
        return true;
    }

    @Override
    public void onLongPress(MotionEvent event) {
        Log.d(DEBUG_TAG, "onLongPress: " + event.toString());
    }

    @Override
    public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX,
            float distanceY) {
        Log.d(DEBUG_TAG, "onScroll: " + e1.toString()+e2.toString());
        return true;
    }

    @Override
    public void onShowPress(MotionEvent event) {
        Log.d(DEBUG_TAG, "onShowPress: " + event.toString());
    }

    @Override
    public boolean onSingleTapUp(MotionEvent event) {
        Log.d(DEBUG_TAG, "onSingleTapUp: " + event.toString());
        return true;
    }

    @Override
    public boolean onDoubleTap(MotionEvent event) {
        Log.d(DEBUG_TAG, "onDoubleTap: " + event.toString());
        return true;
    }

    @Override
    public boolean onDoubleTapEvent(MotionEvent event) {
        Log.d(DEBUG_TAG, "onDoubleTapEvent: " + event.toString());
        return true;
    }

    @Override
    public boolean onSingleTapConfirmed(MotionEvent event) {
        Log.d(DEBUG_TAG, "onSingleTapConfirmed: " + event.toString());
        return true;
    }
}



//監(jiān)測(cè)部分手勢(shì)
public class MainActivity extends Activity {

    private GestureDetectorCompat mDetector;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mDetector = new GestureDetectorCompat(this, new MyGestureListener());
    }

    @Override
    public boolean onTouchEvent(MotionEvent event){
        this.mDetector.onTouchEvent(event);
        return super.onTouchEvent(event);
    }

    class MyGestureListener extends GestureDetector.SimpleOnGestureListener {
        private static final String DEBUG_TAG = "Gestures";

        @Override
        public boolean onDown(MotionEvent event) {
            Log.d(DEBUG_TAG,"onDown: " + event.toString());
            return true;
        }

        @Override
        public boolean onFling(MotionEvent event1, MotionEvent event2,
                float velocityX, float velocityY) {
            Log.d(DEBUG_TAG, "onFling: " + event1.toString()+event2.toString());
            return true;
        }
    }
}

Scroller

彈性滑動(dòng)對(duì)象薄声,用于實(shí)現(xiàn)View的彈性滑動(dòng)当船。

學(xué)習(xí)資料:
1.Android Scroller完全解析,關(guān)于Scroller你所需知道的一切
2.Animating a Scroll Gesture

Scroller的基本用法:

  • 1.創(chuàng)建Scroller實(shí)例
  • 2.調(diào)用startScroll()方法來(lái)初始化
  • 3.重寫View的computeScroll()方法默辨,并在其內(nèi)部完成平滑移動(dòng)的邏輯德频。

View的滑動(dòng)

三種方式實(shí)現(xiàn)View的滑動(dòng):

  • 1.View的ScrollToScrollBy方法。
  • 2.動(dòng)畫
  • 3.改變View的LayoutParams使View重新布局從而實(shí)現(xiàn)滑動(dòng)

使用ScrollTo和ScrollBy

  • ScrollTo相對(duì)于初始位置滾動(dòng)某段距離缩幸。
  • ScrollBy相對(duì)于當(dāng)前位置滾動(dòng)某段距離壹置。
  • 其中,水平方向表谊,正值為向左滾動(dòng)钞护,負(fù)值為向右滾動(dòng)。豎直方向铃肯,正值為向上滾動(dòng)患亿,負(fù)值為向下滾動(dòng)。
  • 兩個(gè)Scroll方法的滾動(dòng)都是跳躍式的,沒(méi)有平滑移動(dòng)的效果步藕。

使用動(dòng)畫

使用動(dòng)畫操作View的translationXtranslationY兩個(gè)屬性來(lái)移動(dòng)View惦界。可使用屬性動(dòng)畫/View動(dòng)畫咙冗。

View動(dòng)畫是對(duì)View的影像進(jìn)行操作的沾歪。也就是說(shuō)View動(dòng)畫并不能真正的改變View的位置。
屬性動(dòng)畫是真正改變View的位置雾消,但它是從Android3.0開(kāi)始推出的灾搏。

//View動(dòng)畫實(shí)現(xiàn)移動(dòng)
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:fillAfter="true"
    >
   <translate
       android:fromXDelta="0"
       android:fromYDelta="0"
       android:toXDelta="500"
       android:toYDelta="1000"
       android:duration="10000"
       android:interpolator="@android:anim/linear_interpolator"
       />
</set>

 Animation animation = AnimationUtils.loadAnimation(this, R.anim.translate);
        layout.setAnimation(animation);
        animation.start();

//使用屬性動(dòng)畫實(shí)現(xiàn)移動(dòng)
ObjectAnimator.ofFloat(layout,"translationX",0,100).setDuration(1000).start();

改變布局參數(shù)

改變布局參數(shù)LayoutParams

 ViewGroup.MarginLayoutParams params = (ViewGroup.MarginLayoutParams) layout.getLayoutParams();
                params.width+=1000;
                params.leftMargin+=1000;
                layout.requestLayout();

三種滑動(dòng)方式的對(duì)比

方式 優(yōu)點(diǎn) 缺點(diǎn)
scrollTo/scrollBy 原生方法立润,操作簡(jiǎn)單狂窑,適用于對(duì)內(nèi)容的滑動(dòng) 不能滑動(dòng)View本身
動(dòng)畫 操作簡(jiǎn)單,適用于沒(méi)有交互的View和復(fù)雜的動(dòng)畫效果 Android3.0以下的兼容問(wèn)題
改變布局參數(shù) 適用于有交互的View 操作較復(fù)雜

彈性滑動(dòng)

將一次時(shí)間長(zhǎng)的滑動(dòng)分成若干次短的滑動(dòng)桑腮。

使用Scroller

//1.創(chuàng)建Scroller
Scroller scroller = new Scroller(context);
...

//2.調(diào)用Scroller的startScroll方法
int scrollX = getScrollX();
int deltaX = destX - scrollX;
scroller.startScroll(scrollX,0,deltaX,0,1000);
invalidate();

//3.復(fù)寫computeScroll()方法
@Override
public void computeScroll(){
    if(scroller.computeScrollOffset()){
       scrollTo(scroller.getCurrX(),scroller.getCurrY());
       invalidate();
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末泉哈,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子破讨,更是在濱河造成了極大的恐慌丛晦,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,692評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件提陶,死亡現(xiàn)場(chǎng)離奇詭異烫沙,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)隙笆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,482評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門锌蓄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人仲器,你說(shuō)我怎么就攤上這事煤率。” “怎么了乏冀?”我有些...
    開(kāi)封第一講書人閱讀 162,995評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵蝶糯,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我辆沦,道長(zhǎng)昼捍,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 58,223評(píng)論 1 292
  • 正文 為了忘掉前任肢扯,我火速辦了婚禮妒茬,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘蔚晨。我一直安慰自己乍钻,他們只是感情好肛循,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,245評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著银择,像睡著了一般多糠。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上浩考,一...
    開(kāi)封第一講書人閱讀 51,208評(píng)論 1 299
  • 那天夹孔,我揣著相機(jī)與錄音,去河邊找鬼析孽。 笑死搭伤,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的袜瞬。 我是一名探鬼主播怜俐,決...
    沈念sama閱讀 40,091評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼吞滞!你這毒婦竟也來(lái)了佑菩?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 38,929評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤裁赠,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后赴精,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體佩捞,經(jīng)...
    沈念sama閱讀 45,346評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,570評(píng)論 2 333
  • 正文 我和宋清朗相戀三年蕾哟,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了一忱。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,739評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡谭确,死狀恐怖帘营,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情逐哈,我是刑警寧澤芬迄,帶...
    沈念sama閱讀 35,437評(píng)論 5 344
  • 正文 年R本政府宣布,位于F島的核電站昂秃,受9級(jí)特大地震影響禀梳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜肠骆,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,037評(píng)論 3 326
  • 文/蒙蒙 一算途、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蚀腿,春花似錦嘴瓤、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,677評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)筛谚。三九已至,卻和暖如春狞贱,著一層夾襖步出監(jiān)牢的瞬間刻获,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,833評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工瞎嬉, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蝎毡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,760評(píng)論 2 369
  • 正文 我出身青樓氧枣,卻偏偏與公主長(zhǎng)得像沐兵,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子便监,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,647評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容