欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android自定義View實(shí)現(xiàn)角度選擇器

 更新時(shí)間:2016年11月06日 11:26:26   投稿:daisy  
前幾天在Google Photos查看照片,用了一下它的圖片剪裁功能,于是我馬上就被其界面和操作吸引。后來想模仿做一個(gè)和Google Photos裁圖頁(yè)面幾乎一模一樣的角度選擇器,本文比較基礎(chǔ),在閱讀本文前只需要掌握最基礎(chǔ)的自定義View知識(shí)和Android事件知識(shí)。下面來一起學(xué)習(xí)下吧。

首先來看一下Google Photos的效果

實(shí)現(xiàn)最終的效果:

實(shí)現(xiàn)思路

仔細(xì)觀察這個(gè)效果,先分析構(gòu)成結(jié)構(gòu),我把它分成三部分:

    1、表示刻度的點(diǎn)

    2、相應(yīng)點(diǎn)上方的數(shù)字

    3、控件中央的當(dāng)前刻度與三角

可以看出,構(gòu)成元素十分簡(jiǎn)單,不涉及圖片,Drawable,那么只需要用Canvas畫出來就好了。

接下來觀察手勢(shì)的操作,查看隨著手指滑動(dòng),控件做出的變化,這里的變化有:

     1、手指按上去時(shí),部分區(qū)域變亮(部分區(qū)域即為可見區(qū)域)

     2、隨著手指滑動(dòng),相應(yīng)的數(shù)字發(fā)生移動(dòng),當(dāng)前角度值也發(fā)生改變

     3、離中心越近,透明度越低,且0°的下方的點(diǎn)要大一些

好了,我們對(duì)這個(gè)控件已經(jīng)分析的很透徹了,根據(jù)分析,首先我們要在View的onDraw()方法中畫出構(gòu)成元素,之后要讓它動(dòng)起來,在onTouchEvent()方法中監(jiān)聽手勢(shì),改變一些值并重繪我們的View,這里很明顯,我們要改變的肯定是當(dāng)前角度這個(gè)值。

動(dòng)手

既然有了思路,那么就要馬上動(dòng)手,不然下次忘記掉了怎么辦?

畫點(diǎn)

首先,先把點(diǎn)給畫出來,位置很簡(jiǎn)單,肯定是從視圖中心開始畫,往左往右分別畫點(diǎn)。

for (int i = 0; i < mPointCount; i++) {
 canvas.getClipBounds(mCanvasClipBounds);
 canvas.drawPoint(mCanvasClipBounds.centerX() + (i - mPointCount / 2) * mPointMargin,
     mCanvasClipBounds.centerY(), mPointPaint);
}

其中mPointCount為所要畫的點(diǎn)個(gè)數(shù),這里默認(rèn)為51個(gè),mPointMargin為兩點(diǎn)間的邊距,長(zhǎng)度為View的寬度除以點(diǎn)的個(gè)數(shù)。

看看效果

嗯,成功的把點(diǎn)給畫出來了。

畫刻度上方的數(shù)字

既然把點(diǎn)給畫出來了,根據(jù)我們的思路,我們要把數(shù)字給畫到正確的位置上,畫數(shù)字當(dāng)然很簡(jiǎn)單,這里難的是找到正確的位置。

首先,我們的數(shù)字是會(huì)移動(dòng)的,隨著當(dāng)前角度的不同,所展示的數(shù)字大小位置都不同。但毫無疑問的是,這個(gè)位置(x坐標(biāo))肯定是關(guān)于當(dāng)前角度的一個(gè)函數(shù)。至于具體是一個(gè)什么樣的函數(shù),相信聰明的你很快就可以分析出來,畢竟只是一個(gè)線性關(guān)系,這里就直接貼代碼了。

private void drawDegreeText(int degrees, Canvas canvas, boolean canReach) {
 canvas.drawText(degrees + "°",
    getWidth() / 2 + mPointMargin * degrees / 2 - mTextWidths[0] / 2 * 3 - mCurrentDegrees / 2 * mPointMargin,
     getHeight() / 2 - 10,
     mTextPaint);
 }
}

按照我們的效果,我們需要畫出-90°~90°的刻度,其中-45°~45°是可到達(dá)角度,另外的角度不可到達(dá)。

drawDegreeText(0, canvas, true);
drawDegreeText(15, canvas, true);
drawDegreeText(30, canvas, true);
drawDegreeText(45, canvas, true);
drawDegreeText(-15, canvas, true);
drawDegreeText(-30, canvas, true);
drawDegreeText(-45, canvas, true);

drawDegreeText(60, canvas, false);
drawDegreeText(75, canvas, false);
drawDegreeText(90, canvas, false);
drawDegreeText(-60, canvas, false);
drawDegreeText(-75, canvas, false);
drawDegreeText(-90, canvas, false);

好了,來看一下效果,可以看到,數(shù)字被畫在了正確的位置。

畫當(dāng)前角度

這個(gè)超級(jí)簡(jiǎn)單呢,位置也特別好確定,上方三角形的Path也非常好知道,但是這里要注意的是,當(dāng)當(dāng)前角度為0°左右時(shí),不應(yīng)該把0°刻度顯示出來。

//畫當(dāng)前角度
if (mCurrentDegrees >= 10) {
 canvas.drawText(mCurrentDegrees + "°", getWidth() / 2 - mTextWidths[0], mBaseLine, mTextPaint);
} else if (mCurrentDegrees <= -10) {
 canvas.drawText(mCurrentDegrees + "°", getWidth() / 2 - mTextWidths[0] / 2 * 3, mBaseLine, mTextPaint);
} else if (mCurrentDegrees < 0) {
 canvas.drawText(mCurrentDegrees + "°", getWidth() / 2 - mTextWidths[0], mBaseLine, mTextPaint);
} else {
 canvas.drawText(mCurrentDegrees + "°", getWidth() / 2 - mTextWidths[0] / 2, mBaseLine, mTextPaint);
}

//三角指示器的Path
mIndicatorPath.moveTo(w / 2, h / 2 + mFontMetrics.top / 2 - 18);
mIndicatorPath.rLineTo(-8, -8);
mIndicatorPath.rLineTo(16, 0);

還有一個(gè)小細(xì)節(jié),就是離中心越近,其透明度越來越低,也就是說,我們要根據(jù)離中心的位置,來改變畫筆Paint的alpha值,再將點(diǎn)畫出。

 for (int i = 0; i < mPointCount; i++) {

  if (i > zeroIndex - 22 && i < zeroIndex + 22 && mScrollStarted) {
   mPointPaint.setAlpha(255);
  } else {
   mPointPaint.setAlpha(100);
  }

  if (i > mPointCount / 2 - 8 && i < mPointCount / 2 + 8
    && i > zeroIndex - 22 && i < zeroIndex + 22) {
   if (mScrollStarted)
    mPointPaint.setAlpha(Math.abs(mPointCount / 2 - i) * 255 / 8);
   else
    mPointPaint.setAlpha(Math.abs(mPointCount / 2 - i) * 100 / 8);
  }
  ……
 }

這時(shí),已經(jīng)很像了,只是還不能動(dòng)。

動(dòng)起來

該繪制的部分我們已經(jīng)都繪制起來了,是時(shí)候讓這個(gè)View動(dòng)起來了,角度選擇器的觸摸事件不復(fù)雜,我們只需要在我們移動(dòng)手指的時(shí)候根據(jù)滑動(dòng)距離來改變當(dāng)前角度值并重繪視圖就可以看到移動(dòng)效果了。為什么呢?因?yàn)槲覀冎霸诋嫈?shù)字的時(shí)候,位置都是由當(dāng)前角度這個(gè)值決定的,所以它一發(fā)生變化,數(shù)字的位置也會(huì)發(fā)生改變。

@Override
public boolean onTouchEvent(MotionEvent event) {
 switch (event.getAction()) {
  ……
  case MotionEvent.ACTION_MOVE:
   float distance = event.getX() - mLastTouchedPosition;
   ……
   if (distance != 0) {
    onScrollEvent(event, distance);
   }
   break;
  }
  ……
  return true;
}

private void onScrollEvent(MotionEvent event, float distance) {
 mTotalScrollDistance -= distance;
 postInvalidate();
 mLastTouchedPosition = event.getX();
 mCurrentDegrees = (int) ((mTotalScrollDistance * mDragFactor) / mPointMargin);
 if (mScrollingListener != null) {
  mScrollingListener.onScroll(mCurrentDegrees);
 }
}

當(dāng)然你還需要處理一些細(xì)節(jié)性的東西,比如在數(shù)字移動(dòng)的時(shí)候,靠近中心一定范圍就會(huì)消失(透明度變?yōu)?),這些都是很容易控制的,只要改變畫筆的透明度就好了,但是正是對(duì)細(xì)節(jié)的把控,才能做出一個(gè)效果讓用戶滿意的自定義View。最后,再來看一下效果。

擴(kuò)展

到這里,我們做的角度選擇器已經(jīng)和Google Photos的幾乎一模一樣了,但是,僅僅這樣就夠了。不,不夠,我們還要繼續(xù)擴(kuò)展,為什么只能達(dá)到正負(fù)45°,我們要所有的范圍自由選擇,也就是-180°~180°,然后數(shù)字顏色啊,點(diǎn)的顏色啊都要讓人自由選擇。所以我們要擴(kuò)展我們的角度選擇器,把一切可以變化的接口暴露出來。

最后看一下我們多種多樣的角度選擇器,還是挺好看呀~

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,這次的自定義View相對(duì)于前兩次的來說,無疑是簡(jiǎn)單了很多,只運(yùn)用了最基礎(chǔ)的繪圖知識(shí)和事件機(jī)制,但是看起來效果也還不錯(cuò)哦,嘿嘿,反正我特別喜歡這個(gè)角度選擇器,雖然我還不知道除了裁圖頁(yè)面可以用到還有哪里可以用到。所以說運(yùn)用最簡(jiǎn)單的知識(shí),也可以組合出比較復(fù)雜的效果,希望大家多發(fā)揮自己的想象力,一起自定義出更多好玩的,實(shí)用的,酷炫的控件吧。希望這篇文章對(duì)你有幫助,哪怕只是一些啟發(fā),也是值得的。如果有疑問大家也可以留言交流。

相關(guān)文章

  • Android實(shí)現(xiàn)拍照、錄像、錄音代碼范例

    Android實(shí)現(xiàn)拍照、錄像、錄音代碼范例

    這篇文章主要介紹了Android實(shí)現(xiàn)拍照、錄像、錄音代碼的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。
    2016-10-10
  • Android消息處理機(jī)制Looper和Handler詳解

    Android消息處理機(jī)制Looper和Handler詳解

    Android應(yīng)用程序是通過消息來驅(qū)動(dòng)的,系統(tǒng)為每一個(gè)應(yīng)用程序維護(hù)一個(gè)消息隊(duì)例,應(yīng)用程序的主線程不斷地從這個(gè)消息隊(duì)例中獲取消息(Looper),然后對(duì)這些消息進(jìn)行處理(Handler),這樣就實(shí)現(xiàn)了通過消息來驅(qū)動(dòng)應(yīng)用程序的執(zhí)行,本文將詳細(xì)分析Android應(yīng)用程序的消息處理機(jī)制
    2014-09-09
  • Android實(shí)現(xiàn)微信的圖片選擇器

    Android實(shí)現(xiàn)微信的圖片選擇器

    這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)微信的圖片選擇器,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • RecyclerView通過GridLayoutManager實(shí)現(xiàn)多樣式布局的示例

    RecyclerView通過GridLayoutManager實(shí)現(xiàn)多樣式布局的示例

    本篇文章主要介紹了RecyclerView通過GridLayoutManager實(shí)現(xiàn)多樣式布局的示例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-12-12
  • android調(diào)用C語言實(shí)現(xiàn)內(nèi)存的讀取與修改的方法示例

    android調(diào)用C語言實(shí)現(xiàn)內(nèi)存的讀取與修改的方法示例

    這篇文章主要介紹了android調(diào)用C語言實(shí)現(xiàn)內(nèi)存的讀取與修改的方法示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-03-03
  • Android給TextView添加點(diǎn)擊事件的實(shí)現(xiàn)方法

    Android給TextView添加點(diǎn)擊事件的實(shí)現(xiàn)方法

    下面小編就為大家?guī)硪黄狝ndroid給TextView添加點(diǎn)擊事件的實(shí)現(xiàn)方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2016-12-12
  • Android handle-message的發(fā)送與處理案例詳解

    Android handle-message的發(fā)送與處理案例詳解

    這篇文章主要介紹了Android handle-message的發(fā)送與處理案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • Android注解ButterKnife的基本使用

    Android注解ButterKnife的基本使用

    這篇文章主要介紹了Android注解ButterKnife的基本使用的相關(guān)資料,需要的朋友可以參考下
    2017-01-01
  • Android獲取assets文件夾中的數(shù)據(jù)并寫入SD卡示例

    Android獲取assets文件夾中的數(shù)據(jù)并寫入SD卡示例

    這篇文章主要介紹了Android獲取assets文件夾中的數(shù)據(jù)并寫入SD卡示例,對(duì)初學(xué)Android開發(fā)的朋友來說是一個(gè)很實(shí)用的功能,需要的朋友可以參考下
    2014-07-07
  • Android如何防止apk程序被反編譯(尊重勞動(dòng)成果)

    Android如何防止apk程序被反編譯(尊重勞動(dòng)成果)

    作為Android應(yīng)用開發(fā)者,不得不面對(duì)一個(gè)尷尬的局面,就是自己辛辛苦苦開發(fā)的應(yīng)用可以被別人很輕易的就反編譯出來,天下痛苦之事莫過于此啊,本文會(huì)介紹一種防止apk程序被反編譯的方法,感興趣的朋友可以了解下哦
    2013-01-01

最新評(píng)論