仿知乎廣告欄----自定義ImageView嵌入RecyclerView

17年底刷知乎時發(fā)現(xiàn)的新型布局展示,感覺很有創(chuàng)意哟绊。知乎用此來投放廣告册赛,用正常列表Item的高度通過滑動展示整張手機屏幕大小的圖片臀叙。下面是仿做的Gif效果圖:

ad-image-view-demo.gif

實現(xiàn)分析

從效果上我們可以大致看出是一個兩種ItemType的RecyclerView笆豁,而且從知乎上來看這個特殊的Item只放了圖片厌丑,所以重點工作就是寫一個自定義ImageView,需要的功能包括展示圖片渔呵,滑動圖片等,現(xiàn)在砍鸠,我們詳細分析下滑動流程扩氢,下圖是滑動過程的邏輯分析:

知乎廣告欄滑動解析圖.png

代碼解讀

自定義View中的核心代碼:

private Context mContext;
/**
 * 展示圖片的寬高和Item高度
 */
private int mDisplayWidth, mDisplayHeight, mItemHeight;
/**
 * Item與背景圖片滑動距離的相對比例
 */
private float ratio = 1;
/**
 * 圖片的滑動區(qū)域
 */
private int mDisplayDistance;
/**
 * 圖片資源
 */
private Drawable imageDrawable;
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    mItemHeight = h;
    imageDrawable = getDrawable();
    // 使用Item的寬度作為圖片的寬度
    mDisplayWidth = getWidth();
    // 可以直接用手機屏幕的高度(別忘了減去狀態(tài)欄高度),不過滑動起來會相對靜止爷辱,比較生硬录豺,可以稍微減少此值以添加相對滑動的感覺
    mDisplayHeight = getScreenHeight(mContext) - getStatusBarHeight(mContext) - 100;
    // 圖片滑動區(qū)域
    mDisplayDistance = mDisplayHeight - mItemHeight;
}

在onDraw中進行狀態(tài)保存和位移

@Override
protected void onDraw(Canvas canvas) {
    imageDrawable.setBounds(0, 0, mDisplayWidth, mDisplayHeight);
    canvas.save();
    canvas.translate(0, -mDisplayDistance * ratio);
    Log.d("SwipeDisplayImageView", "translateY = " + -mDisplayDistance *  ratio);
    super.onDraw(canvas);
    canvas.restore();
}

在RecyclerView滑動監(jiān)聽中調(diào)用朦肘,用來刷新Item在圖片的位置比例,核心邏輯

/**
 * @param itemTop            itemView.getTop()
 * @param recyclerViewHeight recyclerView.getHeight()
 */
public void refreshRatio(int itemTop, int recyclerViewHeight) {
    // 背景圖片位移的范圍
    int scope = recyclerViewHeight - mItemHeight;
    ratio = itemTop * 1.0f / scope;
    Log.d("SwipeDisplayImageView", "ratio = " + ratio);
    if (ratio < 0) {
        ratio = 0;
    }
    if (ratio > 1) {
        ratio = 1;
    }
    invalidate();
}

RecyclerView添加滑動監(jiān)聽

mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
    @Overrid
    public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
        super.onScrolled(recyclerView, dx, dy);
        int firstPosition = mLinearLayoutManager.findFirstCompletelyVisibleItemPosition();
        int lastPosition = mLinearLayoutManager.findLastCompletelyVisibleItemPosition();
        // 邏輯處理的范圍是從Item完全進入到完全離開
        for (int i = firstPosition; i <= lastPosition; i++) {
            RecyclerView.ViewHolder viewHolder = mRecyclerView.findViewHolderForAdapterPosition(i);
            // 判斷ItemType
            if (viewHolder.getItemViewType() == TYPE_AD) {
                View itemView = viewHolder.itemView;
                int top = itemView.getTop();
                int height = recyclerView.getHeight();

                SwipeDisplayImageView swipeDisplayImageView = itemView.findViewById(R.id.iv_swipe_display);
                swipeDisplayImageView.refreshSwipeRatio(top, height);
            }
        }
    }
});

xml代碼

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="180dp">

    <com.example.zengsheng.advertisement.SwipeDisplayImageView
        android:id="@+id/iv_swipe_display"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:scaleType="matrix"
        android:src="@drawable/pic1" />

</RelativeLayout>

總結(jié)

總體來說代碼量不多双饥,沒有自定義繪制的部分媒抠,主要是需要分析清楚效果的實現(xiàn)過程,完成位移的邏輯處理即可咏花,注意在xml中添加android:scaleType="matrix"這條屬性趴生。

github源碼鏈接

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市昏翰,隨后出現(xiàn)的幾起案子苍匆,更是在濱河造成了極大的恐慌,老刑警劉巖棚菊,帶你破解...
    沈念sama閱讀 222,865評論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件浸踩,死亡現(xiàn)場離奇詭異,居然都是意外死亡统求,警方通過查閱死者的電腦和手機检碗,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,296評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來码邻,“玉大人折剃,你說我怎么就攤上這事∶疤玻” “怎么了微驶?”我有些...
    開封第一講書人閱讀 169,631評論 0 364
  • 文/不壞的土叔 我叫張陵铸豁,是天一觀的道長皆辽。 經(jīng)常有香客問我,道長蛤育,這世上最難降的妖魔是什么篇恒? 我笑而不...
    開封第一講書人閱讀 60,199評論 1 300
  • 正文 為了忘掉前任扶檐,我火速辦了婚禮,結(jié)果婚禮上胁艰,老公的妹妹穿的比我還像新娘款筑。我一直安慰自己,他們只是感情好腾么,可當我...
    茶點故事閱讀 69,196評論 6 398
  • 文/花漫 我一把揭開白布奈梳。 她就那樣靜靜地躺著,像睡著了一般解虱。 火紅的嫁衣襯著肌膚如雪攘须。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,793評論 1 314
  • 那天殴泰,我揣著相機與錄音于宙,去河邊找鬼浮驳。 笑死,一個胖子當著我的面吹牛捞魁,可吹牛的內(nèi)容都是我干的至会。 我是一名探鬼主播,決...
    沈念sama閱讀 41,221評論 3 423
  • 文/蒼蘭香墨 我猛地睜開眼谱俭,長吁一口氣:“原來是場噩夢啊……” “哼奉件!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起旺上,我...
    開封第一講書人閱讀 40,174評論 0 277
  • 序言:老撾萬榮一對情侶失蹤瓶蚂,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后宣吱,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體窃这,經(jīng)...
    沈念sama閱讀 46,699評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,770評論 3 343
  • 正文 我和宋清朗相戀三年征候,在試婚紗的時候發(fā)現(xiàn)自己被綠了杭攻。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,918評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡疤坝,死狀恐怖兆解,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情跑揉,我是刑警寧澤锅睛,帶...
    沈念sama閱讀 36,573評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站历谍,受9級特大地震影響现拒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜望侈,卻給世界環(huán)境...
    茶點故事閱讀 42,255評論 3 336
  • 文/蒙蒙 一印蔬、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧脱衙,春花似錦侥猬、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,749評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至荤胁,卻和暖如春铜邮,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,862評論 1 274
  • 我被黑心中介騙來泰國打工松蒜, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人已旧。 一個月前我還...
    沈念sama閱讀 49,364評論 3 379
  • 正文 我出身青樓秸苗,卻偏偏與公主長得像,于是被迫代替她去往敵國和親运褪。 傳聞我的和親對象是個殘疾皇子惊楼,可洞房花燭夜當晚...
    茶點故事閱讀 45,926評論 2 361

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