【Android】View 的滑動

View 的滑動是 Android 實現自定義控件的基礎,同時在開發中我們也難免會遇到 View 的滑動處理。其實不管是哪種滑動方式,其基本思想都是類似的:當點擊事件傳到 View 時,系統記下觸摸點的坐標,手指移動時系統記下移動后觸摸的坐標并算出偏移量,并通過偏移量來修改View的坐標。

實現 View 滑動有很多種方法,在這里主要講解6種滑動方法,分別是 layout()、offsetLeftAndRight() 與 offsetTopAndBottom()、LayoutParams、Animation、scollTo() 與 scollBy(),以及 Scroller。

一、layout() 方法

View 進行繪制的時候會調用 onLayout() 方法來設置顯示的位置,因此我們同樣也可以通過修改 View 的 left、top、right、bottom 這4種屬性來控制 View 的坐標。首先我們要自定義一個 View,在 onTouchEvent() 方法中獲取觸摸點的坐標,代碼如下所示:

override fun onTouchEvent(event: MotionEvent?): Boolean {// 獲取手指觸摸點的橫坐標和縱坐標val x = event?.x?.toInt()val y = event?.y?.toInt()when (event?.action) {MotionEvent.ACTION_DOWN -> {lastX = x ?: 0lastY = y ?: 0}...}...
}

接下來我們在 ACTION_MOVE 事件中計算偏移量,再調用 layout() 方法重新放置這個自定義 View 的位置即可。

override fun onTouchEvent(event: MotionEvent?): Boolean {...when (event?.action) {...MotionEvent.ACTION_MOVE -> {// 計算移動的距離val offsetX = x ?: (0 - lastX)val offsetY = y ?: (0 - lastY)// 調用 layout 方法來重新放置它的位置layout(left + offsetX, top + offsetY, right + offsetX, bottom + offsetY)}}...
}

在每次移動時都會調用 layout() 方法對屏幕重新布局,從而達到移動 View 的效果。自定義 View 的全部代碼如下所示:

class CustomView(context: Context, attributeSet: AttributeSet) : View(context, attributeSet) {private var lastX = 0private var lastY = 0override fun onTouchEvent(event: MotionEvent?): Boolean {// 獲取手指觸摸點的橫坐標和縱坐標val x = event?.x?.toInt()val y = event?.y?.toInt()when (event?.action) {MotionEvent.ACTION_DOWN -> {lastX = x ?: 0lastY = y ?: 0}MotionEvent.ACTION_MOVE -> {// 計算移動的距離val offsetX = x ?: (0 - lastX)val offsetY = y ?: (0 - lastY)// 調用 layout 方法來重新放置它的位置layout(left + offsetX, top + offsetY, right + offsetX, bottom + offsetY)}}return true}
}

隨后,我們在布局中引用自定義 View 就可以了:

<com.tyhoo.android.demo.CustomViewandroid:id="@+id/test_view"android:layout_width="100dp"android:layout_height="100dp"android:background="@android:color/holo_red_light"... />

運行程序,效果如圖1所示:
請添加圖片描述

圖1

圖1中的方塊就是我們自定義的 View,它會隨著我們手指的滑動改變自己的位置。

二、offsetLeftAndRight() 與 offsetTopAndBottom()

這兩種方法和 layout() 方法的效果差不多,其使用方式也差不多。我們將 ACTION_MOVE 中的代碼替換成如下代碼:

override fun onTouchEvent(event: MotionEvent?): Boolean {...when (event?.action) {...MotionEvent.ACTION_MOVE -> {// 計算移動的距離val offsetX = x ?: (0 - lastX)val offsetY = y ?: (0 - lastY)// 對 left 和 right 進行偏移offsetLeftAndRight(offsetX)// 對 top 和 bottom 進行偏移offsetTopAndBottom(offsetY)}}...
}

三、LayoutParams

LayoutParams 主要保存了一個 View 的布局參數,因此我們可以通過 LayoutParams 來改變 View 的布局參數從而達到改變 View 位置的效果。同樣,我們將 ACTION_MOVE 中的代碼替換成如下代碼:

override fun onTouchEvent(event: MotionEvent?): Boolean {...when (event?.action) {...MotionEvent.ACTION_MOVE -> {// 計算移動的距離val offsetX = x ?: (0 - lastX)val offsetY = y ?: (0 - lastY)val layoutParams = layoutParams as ConstraintLayout.LayoutParamslayoutParams.leftMargin = left + offsetXlayoutParams.topMargin = top + offsetYsetLayoutParams(layoutParams)}}...
}

因為父控件是 ConstraintLayout,所以我們用了 ConstraintLayout.LayoutParams。如果父控件是 RelativeLayout,則要使用RelativeLayout.LayoutParams。除了使用布局的 LayoutParams 外,我們還可以用 ViewGroup.MarginLayoutParams 來實現:

override fun onTouchEvent(event: MotionEvent?): Boolean {...when (event?.action) {...MotionEvent.ACTION_MOVE -> {// 計算移動的距離val offsetX = x ?: (0 - lastX)val offsetY = y ?: (0 - lastY)val layoutParams = layoutParams as ViewGroup.MarginLayoutParamslayoutParams.leftMargin = left + offsetXlayoutParams.topMargin = top + offsetYsetLayoutParams(layoutParams)}}...
}

四、Animation

可以采用 View 動畫來移動,在 res 目錄新建 anim 文件夾并創建 translate.xml:

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"><translateandroid:duration="1000"android:fromXDelta="0"android:toXDelta="300" />
</set>

接下來在 Kotlin 代碼中調用就好了,代碼如下所示:

class MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val testView = findViewById<CustomView>(R.id.test_view)testView.animation = AnimationUtils.loadAnimation(this, R.anim.translate)}
}

運行程序,效果如圖2所示:
請添加圖片描述

圖2

運行程序,我們設置的方塊會向右平移300像素,然后又會回到原來的位置。為了解決這個問題,我們需要在 translate.xml 中加上 fillAfter=“true”,代碼如下所示:

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"android:fillAfter="true"><translateandroid:duration="1000"android:fromXDelta="0"android:toXDelta="300" />
</set>

運行程序,效果如圖3所示:
請添加圖片描述

圖3

運行代碼后會發現,方塊向右平移300像素后就停留在當前位置了。

需要注意的是,View 動畫并不能改變 View 的位置參數。如果對一個 View 進行如上的平移動畫操作,當 View 平移300像素停留在當前位置時,我們點擊這個 View 并不會觸發點擊事件,但在我們點擊這個 View 的原始位置時卻觸發了點擊事件。對于系統來說這個 View 并沒有改變原有的位置,所以我們點擊其他位置當然不會觸發這個 View 的點擊事件。

五、scrollTo() 與 scollBy()

scrollTo(x, y) 表示移動到一個具體的坐標點,而 scrollBy(dx, dy) 則表示移動的增量為 dx、dy。其中,scollBy 最終也是要調用 scollTo 的。View 的 scollTo 和 scollBy 的源碼如下所示:

public void scrollTo(int x, int y) {if (mScrollX != x || mScrollY != y) {int oldX = mScrollX;int oldY = mScrollY;mScrollX = x;mScrollY = y;invalidateParentCaches();onScrollChanged(mScrollX, mScrollY, oldX, oldY);if (!awakenScrollBars()) {postInvalidateOnAnimation();}}
}public void scrollBy(int x, int y) {scrollTo(mScrollX + x, mScrollY + y);
}

scollTo、scollBy 移動的是 View 的內容,如果在 ViewGroup 中使用,則是移動其所有的子 View。我們將 ACTION_MOVE 中的代碼替換成如下代碼:

override fun onTouchEvent(event: MotionEvent?): Boolean {...when (event?.action) {...MotionEvent.ACTION_MOVE -> {// 計算移動的距離val offsetX = x ?: (0 - lastX)val offsetY = y ?: (0 - lastY)(parent as View).scrollBy(-offsetX, -offsetY)}}return true
}

這里若要實現自定義 View 隨手指移動的效果,就需要將偏移量設置為負值。為什么要設置為負值呢?這是參考對象不同導致的差異。所以我們用 scrollBy 方法的時候要設置負數才會達到自己想要的效果。

六、Scroller

我們在用 scollTo/scollBy 方法進行滑動時,這個過程是瞬間完成的,所以用戶體驗不大好。這里我們可以使用 Scroller 來實現有過渡效果的滑動,這個過程不是瞬間完成的,而是在一定的時間間隔內完成的。Scroller 本身是不能實現 View 的滑動的,它需要與 View 的 computeScroll() 方法配合才能實現彈性滑動的效果。在這里我們實現自定義 View 平滑地向右移動。首先我們要初始化 Scroller,代碼如下所示:

class CustomView(context: Context, attributeSet: AttributeSet) : View(context, attributeSet) {...private var scroller: Scroller? = nullinit {scroller = Scroller(context)}...
}

接下來重寫 computeScroll() 方法,系統會在繪制 View 的時候在 draw() 方法中調用該方法。在這個方法中,我們調用父類的 scrollTo() 方法并通過 Scroller 來不斷獲取當前的滾動值,每滑動一小段距離我們就調用invalidate() 方法不斷地進行重繪,重繪就會調用 computeScroll() 方法,這樣我們通過不斷地移動一個小的距離并連貫起來就實現了平滑移動的效果。

override fun computeScroll() {super.computeScroll()scroller?.let {if (it.computeScrollOffset()) {(parent as View).scrollTo(it.currX, it.currY)invalidate()}}
}

我們在自定義 View 中寫一個 smoothScrollTo 方法,調用 Scroller 的 startScroll() 方法,在 2000ms 內沿 X 軸平移 delta 像素,代碼如下所示:

fun smoothScrollTo(destX: Int, destY: Int) {val scrollX = scrollXval delta = destX - scrollXscroller?.startScroll(scrollX, 0, delta, 0, 2000)invalidate()
}

最后我們再調用自定義 View 的 smoothScrollTo() 方法。這里我們設定自定義 View 沿著 X 軸向右平移 400 像素。

class MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val testView = findViewById<CustomView>(R.id.test_view)testView.smoothScrollTo(-400, 0)}
}

運行程序,效果如圖4所示:
請添加圖片描述

圖4

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/710635.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/710635.shtml
英文地址,請注明出處:http://en.pswp.cn/news/710635.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

【AI+應用】怎么快速制作一個類chatGPT套殼網站

最近有人問我&#xff0c; 看了我之前寫的一篇文章 [人工智能] AI浪潮下Sora對于普通人的機會 &#xff0c; 怎么做一個類chatGPT的套殼網站&#xff0c;是從0開始做么。 對于普通人來說&#xff0c;萬事不懂先AI&#xff0c; AI找不到答案搜索google或百度。對于程序員來說…

C# 獲取類型 Type.GetType()

背景 C#是強類型語言&#xff0c;任何對象都有Type&#xff0c;有時候需要使用Type來進行反射、序列化、篩選等&#xff0c;獲取Type有Type.GetType, typeof()&#xff0c;object.GetType() 等方法&#xff0c;本文重點介紹Type.GetType()。 系統類型/本程序集內的類型 對于系…

有哪些視頻媒體?邀請視頻媒體報道活動的好處

傳媒如春雨&#xff0c;潤物細無聲&#xff0c;大家好&#xff0c;我是51媒體網胡老師。 視頻媒體在當今的媒體生態中占據了重要的地位。以下是一些主要的視頻媒體類型&#xff1a; 電視臺&#xff1a;如中央電視臺、各省級衛視臺、地方電視臺等&#xff0c;他們擁有專業的視…

學習linux從0到初級工程師-3

一、LNMP 1.1 搭建LNMP LNMP&#xff1a;LinuxNginxMysqlPHP LNMP優勢&#xff1a; 1.web服務器一種&#xff0c;Nginx處理靜態文件、索引文件&#xff0c;自動索引的效率非常高&#xff1b; 2.作為代理服務器,Nginx可以實現無緩存的反向代理加速&#xff0c;提高網站運行…

探索Redis 6.0的新特性

Redis&#xff08;Remote Dictionary Server&#xff09;是一個開源的內存中數據結構存儲系統&#xff0c;通常被用作緩存、消息隊列和實時數據處理等場景。它的簡單性、高性能以及豐富的數據結構支持使其成為了眾多開發者和企業的首選。在Redis 6.0版本中&#xff0c;引入了一…

Vue3報錯Promise executor functions should not be async.

解決方法 加注釋。。。// eslint-disable-next-line no-async-promise-executor // eslint-disable-next-line no-async-promise-executor new Promise<boolean>(async (resolve, reject) > {... }),

Ubuntu綁定USB接口到固定端口

綁定端口 打開終端&#xff0c;輸入以下命令查看USB端口信息&#xff1a; udevadm info -a -n /dev/ttyUSB0執行后&#xff0c;可以看到部分輸出如下: 找到第一個&#xff0c;a-b:c格式的KERNELS&#xff0c;記住這個值&#xff0c;后面會用到。 linlin-B660M-D2H-DDR4:~$ u…

【深藍學院】移動機器人運動規劃--第7章 集群機器人運動規劃--筆記

文章目錄 0. Contents1. Multi-Agent Path Finding (MAPF)1.1 HCA*1.2 Single-Agent A*1.3 ID1.4 M*1.5 Conflict-Based Search(CBS)1.6 ECBS1.6.1 heuristics1.6.2 Focal Search 2. Velocity Obstacle (VO&#xff0c;速度障礙物)2.1 VO2.2. RVO2.3 ORCA 3. Flocking model&am…

【每日前端面經】2023-02-29

題目來源: 牛客 如何理解前端這個崗位 簡單地說就是設計師做好網頁效果圖&#xff0c;前端將效果圖轉化成頁面&#xff0c;之后交給后端程序員&#xff0c;中間的這段工作就是前端 瀏覽器如何渲染HTML 將載入的HTML文件解析成DOM樹&#xff0c;并且將各個標記標識解析成DOM…

SQL的窗口函數

SQL的窗口函數 文章目錄 SQL的窗口函數1. 介紹2. 聚合函數0.數據準備1. AVG2. COUNT3. MAX4. MIN5. 標準差6. SUM 3. 排序函數1. CUME_DIST2. RANK, DENSE_RANK, ROW_NUMBER3. PERCENT_RANK4. NTILE 4. 值函數(偏移函數)1. FIRST_VALUE2. LAST_VALUE3. LAG4. LEAD5. NTH_VALUE …

ChatGPT4.0 的優勢、升級 4.0 為什么這么難以及如何進行升級?

前言 “ChatGPT4.0一個月多少人民幣&#xff1f;” ”chatgpt4賬號“ ”chatgpt4 價格“ “chatgpt4多少錢” 最近發現很多小伙伴很想知道關于ChatGPT4.0的事情&#xff0c;于是寫了這篇帖子&#xff0c;幫大家分析一下。 一、ChatGPT4.0 的優勢 &#xff08;PS&#xff1a;…

LINUX基礎培訓二十七之shell標準輸入、輸出、錯誤

一、Shell 輸入/輸出重定向 大多數 UNIX 系統命令從你的終端接受輸入并將所產生的輸出發送回??到您的終端。一個命令通常從一個叫標準輸入的地方讀取輸入&#xff0c;默認情況下&#xff0c;這恰好是你的終端。同樣&#xff0c;一個命令通常將其輸出寫入到標準輸出&#xff…

【樹莓派系統配置+python3.8+環境配置踩坑點匯總】raspberrypi

最近又開始搞樹莓派的深度學習模型。很多windows端的環境需要在樹莓派上重新部署&#xff0c;中間出現了非常多的問題。主要以各種庫的下載安裝為主要。 首先&#xff0c;第一個問題&#xff1a; 樹莓派系統燒錄之后&#xff0c;默認apt一般需要升級看&#xff0c;而默認下載…

無窮級數法求Π

任務描述 本關任務&#xff1a;編寫一個無窮級數法計算圓周率的小程序。 相關知識 為了完成本關任務&#xff0c;你需要掌握&#xff1a; 無窮級數法 無窮級數法 π 是個超越數&#xff0c;圓周率的超越性否定了化圓為方這種尺規作圖精確求解問題的可能性。有趣的是&…

【Spring】18 Bean 定義繼承

文章目錄 介紹聲明式配置抽象Bean定義繼承的配置項注意&#xff1a;抽象Bean預實例化結論 Spring 框架提供了一個強大的功能&#xff0c;稱為 Bean 定義繼承&#xff0c; 允許開發人員高效地在 bean 之間重用和自定義配置。在本篇文章中我們將介紹 Bean 定義繼承的概念&#x…

JVM性能優化

運行時優化 方法內聯 方法內聯&#xff0c;是指 JVM在運行時將調用次數達到一定閾值的方法調用替換為方法體本身 &#xff0c;從而消除調用成本&#xff0c;并為接下來進一步的代碼性能優化提供基礎&#xff0c;是JVM的一個重要優化手段之一。 注&#xff1a; C的inline屬于編…

babylonsjs入門-基礎模版

基于babylonjs封裝的一些功能和插件 &#xff0c;希望有更多的小伙伴一起玩babylonjs&#xff1b; 歡迎加群&#xff08;點擊群號傳送&#xff09;&#xff1a;464146715 官方文檔 中文文檔 案例傳送門 ? 懶得打字&#xff0c;你們直接去copy組件吧&#xff0c;主要看這2…

舊版android模擬器,37歲程序員被裁

前言 從18年畢業至今&#xff0c;就職過兩家公司&#xff0c;大大小小項目做了幾個&#xff0c;非常感謝我的兩位老大&#xff0c;在我的android成長路上給予我很多指導&#xff0c;亦師亦友的關系。 從年前至今參加面試了很多公司&#xff0c;也收到了幾家巨頭的offer&#…

transformer--編碼器2(前饋全連接層、規范化層、子層鏈接結構、編碼器層、編碼器)

前饋全連接層 什么是前饋全連接層: 在Transformer中前饋全連接層就是具有兩層線性層的全連接網絡 前饋全連接層的作用: 考慮注意力機制可能對復雜過程的擬合程度不夠,通過增加兩層網絡來增強模型的能力 code # 前饋全連接層 class PositionwiseFeedForward(nn.Module):de…

絕地求生:發現吃雞號被盜,怎么操作才是最正確的

首先閑游盒先了解一下盜號者的盜號流程 一般盜號的流程是先把你steam上的皮膚飾品出售&#xff0c;然后把余額轉走&#xff0c;再把steam賬號作為黑號進行出售。 所以當閑游盒發現號被盜的時候也分為兩種情況&#xff1a;一種是他正在出售商品的時候&#xff0c;你收到郵箱提示…