中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

Android可自定義神奇動效的卡片切換視圖實例

發布時間:2020-08-19 15:32:18 來源:腳本之家 閱讀:306 作者:BakerJ 欄目:移動開發

前言

面對眾多卡片層疊效果,我們的產品童鞋也突發奇想,搞出了另一種卡片層疊切換展示的交互,而且產品狗們居然要求多做幾種動效給他們看,好讓他們選擇,這簡直就是要搞事情啊,what are you 弄啥咧?!

“哥哥我做不到啊.....啊.....呸”,做為一名有節操的程序猿,自然是不能說出這么沒有出息的話,哥就滿足你們,于是,出了個可自定義動效的卡片切換視圖,效果如下所示

Android可自定義神奇動效的卡片切換視圖實例

思路

首先,要展示出卡片層疊的視覺效果。在這里,我們通過方塊的縮放大小差異以及在Y方向上的位置差異,來展現這種視覺效果。

其次,要能夠方便的定義卡片視圖內容。我們通過都很熟悉的設置Adapter的方式來構建內容視圖。

最后,要能夠自定義動效。我們通過定義一個由0到1的ValueAnimator,即每個動畫的過程,其實就是該ValueAnimator在一個動畫周期內,從0變化到1的過程,0表示動畫剛開始,1表示動畫結束了,0.5則表示這一輪動畫已經執行到了一半。這樣,通過轉換器以及插值器,我們就可以根據ValueAnimator實時的值,來設置當前正在執行動畫的卡片應該有的“樣子”。

總覽

我們給出三種基本的動畫模式

/*
 * ANIM_TYPE_FRONT:被選中的卡片通過自定義動效移至第一,其他的卡片通過通用動效補位
 * ANIM_TYPE_SWITCH:選中的卡片和第一張卡片互換位置,并都是自定義動效
 * ANIM_TYPE_FRONT_TO_LAST:第一張圖片通過自定義動效移至最后,其他卡片通過通用動效補位
 */
public static final int ANIM_TYPE_FRONT = 0, ANIM_TYPE_SWITCH = 1, ANIM_TYPE_FRONT_TO_LAST = 2;

并通過Helper類來處理所有的動畫邏輯,以及Adapter來生成卡片視圖

private CardAnimationHelper mAnimationHelper;
private BaseAdapter mAdapter;

在onMeasure時根據卡片寬高比來設置卡片的尺寸,在此請注意,當前情況下卡片寬度與整體容器寬度一致,后續通過自定義的方式,通過縮放來產生卡片的視覺效果。

private float mCardRatio = CARD_SIZE_RATIO;//寬高比:卡片高 / 卡片寬
private int mCardWidth, mCardHeight;//卡片寬高
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  ...
  if (mCardWidth == 0 || mCardHeight == 0) {
    setCardSize(true);
  }
}
/*
 * 根據卡片比例計算卡片寬高,并傳入Helper
 */
private void setCardSize(boolean resetAdapter) {
  mCardWidth = getMeasuredWidth();
  mCardHeight = (int) (mCardWidth * mCardRatio);
  mAnimationHelper.setCardSize(mCardWidth, mCardHeight);
  mAnimationHelper.initAdapterView(mAdapter, resetAdapter);
}

那么如此之后,自然Helper中就保存了視圖的主要數據與參數

//卡片列表
private LinkedList<CardItem> mCards;
//當前正在向后以及向前移動的卡片
private CardItem mCardToBack, mCardToFront;
//正在向后以及向前移動卡片的位置
private int mPositionToBack = 0, mPositionToFront = 0;
//動畫運行的ValueAnimator
private ValueAnimator mValueAnimator;
//當前的動畫系數
private float mCurrentFraction = 1;
...//以及一系列的轉換器與插值器

細節

那么,動畫到底是如何實現的,以及如何自定義的呢,我們以通用動畫為例,來看看動畫的主要流程

首先,在ValueAnimator更新的時候,獲得當前的動畫系數,依次來執行動畫

/**
 * ValueAnimator動畫更新
 */
@Override
public void onAnimationUpdate(ValueAnimator animation) {
  //獲取當前動畫系數
  mCurrentFraction = (float) animation.getAnimatedValue();
  //通過插值器獲取插值系數
  fractionInterpolated = mAnimInterpolator.getInterpolation(mCurrentFraction);
  doAnimationCommon(mCurrentFraction, fractionInterpolated);
  ...
}

接著,對需要執行動畫的卡片,執行動畫,以ANIM_TYPE_FRONT動畫模式為例,當選中的卡片移動到最前的時候,原來在這張卡片之前的所有卡片,都要向后移動一位,來留出第一個的位置

/**
 * 執行通用動畫
 */
private void doAnimationCommon(float fraction, float fractionInterpolated) {
  //如果當前動畫模式為選中的卡片移到最前
  if (mAnimType == InfiniteCardView.ANIM_TYPE_FRONT) {
    //則遍歷在選中卡片之前的卡片
    for (int i = 0; i < mPositionToFront; i++) {
      CardItem card = mCards.get(i);
      //對卡片執行動畫,從當前位置移動到后一個位置
      doAnimationCommonView(card.view, fraction, fractionInterpolated, i, i + 1);
      ...
    }
  }...
}

最后,通過轉換器,對卡片進行自定義動畫處理

/**
 * 對視圖執行通用動畫
 * @param view         卡片視圖
 * @param fromPosition     從該位置
 * @param toPosition      移動到該位置
 */
private void doAnimationCommonView(View view, float fraction, float fractionInterpolated, int
    fromPosition, int toPosition) {
  //通用轉換器轉換動畫
  mTransformerCommon.transformAnimation(view, fraction, mCardWidth,
      mCardHeight, fromPosition, toPosition);
  if (mAnimInterpolator != null) {
    //通用轉換器轉換插值動畫
    mTransformerCommon.transformInterpolatedAnimation(view, fractionInterpolated,            mCardWidth,mCardHeight, fromPosition, toPosition);
  }
}

而轉換器的具體實現則以DefaultCommonTransformer為例

@Override
public void transformAnimation(View view, float fraction, int cardWidth, int cardHeight, int fromPosition, int toPosition) {
  //需要跨越的卡片數量
  int positionCount = fromPosition - toPosition;
  //以0.8做為第一張的縮放尺寸,每向后一張縮小0.1
  //(0.8f - 0.1f * fromPosition) = 當前位置的縮放尺寸
  //(0.1f * fraction * positionCount) = 移動過程中需要改變的縮放尺寸
  float scale = (0.8f - 0.1f * fromPosition) + (0.1f * fraction * positionCount);
  ViewHelper.setScaleX(view, scale);
  ViewHelper.setScaleY(view, scale);
  //在Y方向的偏移量,每向后一張,向上偏移卡片寬度的0.02
  //-cardHeight * (0.8f - scale) * 0.5f 對卡片做整體居中處理
  ViewHelper.setTranslationY(view, -cardHeight * (0.8f - scale) * 0.5f - cardWidth * (0.02f *
     fromPosition - 0.02f * fraction * positionCount));
}

對于向第一位移動的選中卡片,也是同理,只不過是根據該卡片對應的轉換器來進行自定義動畫的轉換。

最后的效果,就像演示圖中第一次點擊,圖片向前翻轉到第一位的效果一樣。

對于產品狗突如其來的想法,咱們程序猿不善于口水仗的,就只能用代碼來讓他們來服氣了。畢竟,大家還都是伐木累嘛,哈哈。

當實現某個東西遇到困難時,不妨想想Android系統自身的一些實現方式,比如參考ListView的Adapter,ViewPager定義翻頁動畫的Transformer等等,總會有意想不到的啟發。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對億速云的支持。如果你想了解更多相關內容請查看下面相關鏈接

向AI問一下細節

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

AI

大荔县| 临沭县| 炎陵县| 玉田县| 曲阜市| 景宁| 彭阳县| 固原市| 綦江县| 苍山县| 抚顺市| 武清区| 龙山县| 虎林市| 静宁县| 邵阳县| 碌曲县| 怀来县| 大余县| 舞阳县| 南岸区| 甘德县| 乳源| 寿阳县| 漳平市| 夏邑县| 临武县| 社旗县| 沛县| 和顺县| 芷江| 武清区| 宣汉县| 鄄城县| 休宁县| 保德县| 西林县| 仙游县| 大兴区| 华容县| 顺平县|