91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Android怎么使用cos和sin繪制復合曲線動畫

發布時間:2021-03-17 14:02:47 來源:億速云 閱讀:295 作者:小新 欄目:開發技術

這篇文章將為大家詳細講解有關Android怎么使用cos和sin繪制復合曲線動畫,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

在開發新需求的時候,設計給了一份類似這樣的動畫:

Android怎么使用cos和sin繪制復合曲線動畫

看著不難,即使一遍看不懂,嘿嘿,不還有設計稿。

Android怎么使用cos和sin繪制復合曲線動畫

作為一個平時很少寫動畫的 Android 開發仔,看到一段段的緩入緩出曲線的設計稿時,我的心情是這樣的:

雖然,Android 動畫默認的插值器 AccelerateDecelerateInterpolator 有這樣緩入緩出的效果:

Android怎么使用cos和sin繪制復合曲線動畫

我總不能一整個動畫給它拆成4段動畫來寫,還別說,我第一次寫的代碼還真的是這么干的。

第一次分析

本著能少寫一行絕不多寫一字的原則,詢問了大佬同事的意見,大佬大手一揮:PathInterpolator(后證實有問題)。

簡單看了一下使用方式,需要使用 Path,再看了一眼,好家伙,有可能會用到貝塞爾曲線,放棄~

為了能夠快速的解決問題,就使用了上面談到的方案:

private fun animateTagView(tagView: TextView) {
 // [0,200]區間的動畫 
 val valueAnimatorOne = ValueAnimator.ofInt(0, 200)
 valueAnimatorOne.addUpdateListener {
  val per = it.animatedValue as Int / 200f
  tagView.rotation = 4 * per
  tagView.scaleX = (1 - 0.1 * per).toFloat()
  tagView.scaleY = (1 - 0.1 * per).toFloat()
 }
 valueAnimatorOne.duration = 200
 // [200,560]區間的動畫
 val valueAnimatorTwo = ValueAnimator.ofInt(200, 560)
 valueAnimatorTwo.addUpdateListener {
  val per = (it.animatedValue as Int - 200) / 360f
  tagView.rotation = 3 - 11 * per
  tagView.scaleX = (0.9 + 0.1 * per).toFloat()
  tagView.scaleY = (0.9 + 0.1 * per).toFloat()
 }
 valueAnimatorTwo.duration = 360
 // [560,840]區間的動畫
 val valueAnimatorThree = ValueAnimator.ofInt(560, 840)
 valueAnimatorThree.addUpdateListener {
  val per = (it.animatedValue as Int - 560) / 280f
  tagView.rotation = -8 + 12 * per
  tagView.scaleX = (1 - 0.2 * per).toFloat()
  tagView.scaleY = (1 - 0.2 * per).toFloat()
 }
 valueAnimatorThree.duration = 280
 // [840,1000]的動畫
 val valueAnimatorFour = ValueAnimator.ofInt(840, 1000)
 valueAnimatorFour.addUpdateListener {
  val per = (it.animatedValue as Int - 840) / 160f
  tagView.rotation = 4 - 4 * per
  tagView.scaleX = (0.8 + 0.2 * per).toFloat()
  tagView.scaleY = (0.8 + 0.2 * per).toFloat()
 }
 valueAnimatorFour.duration = 160
 // 使用AnimatorSet串行執行動畫
 val animationSet = AnimatorSet()
 animationSet.playSequentially(valueAnimatorOne, valueAnimatorTwo, valueAnimatorThree, valueAnimatorFour)
 tagView.post {
  tagView.pivotX = 0f
  tagView.pivotY = ad_tag_two.measuredHeight.toFloat()
  animationSet.start()
 }
}

整個動畫被我拆成了[0,200]、[200,560]、[560,840]和[840,1000]四段屬性動畫,因為產品說只需要播放一次,所以使用 AnimatorSet 將動畫組裝起來,就可以解決問題。

第二次分析

第一次得到的方案雖然能夠解決問題,如果遇到循環播放,AnimatorSet 就不行了,有沒有其他方案呢?

趁著周末的時間,學了一下 PathInterpolator,發現這個玩意也解決不了問題,或者說不好解決問題,雖然可以用三階貝塞爾曲線分段畫出上述曲線,但 PathInterpolator 要求起點和終點分別在 (0,0) 和 (1,1)。

既然插值器不行,可以試試估值器,但一個估值器也解決不了旋轉和縮放兩種動畫,看來得靠 AnimatorUpdateListener 去解決問題。

回頭想一下,插值器是將均勻的時間片段轉化成加速或者減速的行為,我們也可以將均勻的時間片段轉化成對應的曲線,只要做好兩點:

使用線性的插值器 LinearInterpolator。
將上面的曲線拆分,通過不同的 sin 或者 cos 方法表達。
以旋轉動畫為例,拆成的 sin 函數:

Android怎么使用cos和sin繪制復合曲線動畫

另外一段動畫的函數可以參考代碼:

override fun onCreate(savedInstanceState: Bundle?) {
 super.onCreate(savedInstanceState)
 setContentView(R.layout.activity_main)

 val tvContent = findViewById<TextView>(R.id.tv_content)
 val valueAnimatorOne = ValueAnimator.ofFloat(0.0f, 1.5f)
 valueAnimatorOne.addUpdateListener {
  // 通過對應的sin和cos設置rotation和scale
  val per = it.animatedValue as Float
  var rotation: Float = 0f
  var scale: Float = 0f
  if(per >= 0 && per < 0.2f){
   rotation = sin((per / 0.2f) * Math.PI.toFloat() - Math.PI.toFloat() / 2) * 1.5f + 1.5f
   scale = cos(per / 0.2f * Math.PI.toFloat()) * 0.05f + 0.95f
  }
  if(per >= 0.2f && per < 0.56f){
   rotation = sin(Math.PI.toFloat() / 2 + Math.PI.toFloat() * ( per - 0.2f) / 0.36f) * 5.5f - 2.5f
   scale = cos((per - 0.2f) / 0.36f * Math.PI.toFloat() + Math.PI.toFloat()) * 0.05f + 0.95f
  }
  if(per >= 0.56f && per < 0.84f){
   rotation = sin(Math.PI.toFloat() * (per - 0.56f) / 0.28f - Math.PI.toFloat() / 2) * 6f - 2f
   scale = cos((per - 0.56f) / 0.28f * Math.PI.toFloat()) * 0.1f + 0.9f
  }
  if(per in 0.84f..1f){
   rotation = sin(Math.PI.toFloat() / 2 + Math.PI.toFloat() * (per - 0.84f) / 0.16f ) * 2f + 2f
   scale = cos((per - 0.84f) / 0.16f * Math.PI.toFloat() + Math.PI.toFloat()) * 0.1f + 0.9f
  }
  // 設置停止時間
  if(per > 1f && per <= 1.5f){
   rotation = 0f
   scale = 1.0f
  }
  tvContent.rotation = rotation
  tvContent.scaleX = scale
  tvContent.scaleY = scale
 }
 // 設置線性插值器
 valueAnimatorOne.interpolator = LinearInterpolator()
 // 動畫時間
 valueAnimatorOne.duration = 1500
 // 無線循環
 valueAnimatorOne.repeatCount = -1
 tvContent.post {
  // 設置中心點
  tvContent.pivotX = 0f
  tvContent.pivotY = tvContent.measuredHeight.toFloat()
  valueAnimatorOne.start()
 }
}

整個代碼還是比較簡單的,旋轉動畫曲線由 sin 得出,縮放由 cos 得出,最后改一下中心點。

關于“Android怎么使用cos和sin繪制復合曲線動畫”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

崇阳县| 娱乐| 绍兴市| 鄂温| 延川县| 视频| 饶平县| 莱西市| 卓资县| 吉木萨尔县| 博罗县| 荔浦县| 鞍山市| 宜州市| 鸡东县| 阆中市| 嘉祥县| 什邡市| 高尔夫| 类乌齐县| 威海市| 乐陵市| 资阳市| 长武县| 三河市| 会昌县| 武城县| 谷城县| 枣阳市| 镇康县| 申扎县| 博爱县| 义乌市| 襄汾县| 休宁县| 辽宁省| 陇西县| 绿春县| 彩票| 普宁市| 康保县|