實(shí)現(xiàn)抖音評(píng)論彈窗視頻聯(lián)動(dòng)縮放效果

有一年多沒打開抖音了皿桑,因最近疫情影響只能在家里待著详拙,閑來無事刷刷抖音叹阔,發(fā)現(xiàn)抖音從最開始的彈窗已換成了彈出彈窗視頻在最上方并且按一定比例縮放质蕉,如下圖

image

要實(shí)現(xiàn)上圖聯(lián)動(dòng)效果势篡,接下來提幾個(gè)方法:

setScaleX(float scaleX) //水平方向的縮放比例
setScaleY(float scaleY) //垂直方向的縮放比例
//scaleX scaleY = 1.0f 表示初始大小
// scaleX scaleY < 1.0f翩肌,表示縮小善玫,如scale=0.5f直颅,表示寬高是原來的0.5倍
//scaleX scaleY> 1.0f,表示放大启上,如scale=2.0f碍侦,表示寬高是原來的2.0倍
//設(shè)置錨點(diǎn)的X坐標(biāo)值粱坤,以像素為單位。默認(rèn)是View的中心瓷产。
setPivotX(float pivotX)
//設(shè)置錨點(diǎn)的Y坐標(biāo)值站玄,以像素為單位。默認(rèn)是View的中心濒旦。
setPivotX(float pivotX)

其他的我就不一一介紹了株旷,本文主要用到的就是這幾個(gè)方法,以下是幾個(gè)方法的介紹:

【Android開發(fā)】View的平移尔邓、縮放灾常、旋轉(zhuǎn)以及位置、坐標(biāo)系_eieihihi的專欄-CSDN博客


奔入主題铃拇。
接下來講一下我的思路。
我在這里并沒有用特別復(fù)雜的功能沈撞,所以本文主要是為了實(shí)現(xiàn)滑動(dòng)縮放功能

首先分析一下布局方式慷荔,主頁一個(gè)視頻View,然后點(diǎn)擊評(píng)論按鈕彈出底部操作欄缠俺。
好显晶,放出布局,一個(gè)視頻一個(gè)按鈕

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

    <VideoView
        android:id="@+id/videoView"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />


    <android.support.design.widget.FloatingActionButton
        android:id="@+id/floatBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentEnd="true"
        android:layout_alignParentBottom="true"
        android:layout_gravity="bottom|end"
        android:layout_margin="@dimen/fab_margin"
        app:srcCompat="@android:drawable/ic_dialog_email" />

</RelativeLayout>

Fab添加點(diǎn)擊事件壹士,彈出BottomSheetDialog

        findViewById(R.id.floatBtn).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                BottomSheetDialog bsd= new BottomSheetDialog();
                bsd.setFragmentManager(getSupportFragmentManager())
                        .setLayoutRes(R.layout.view_dialog_comment)
                        .setCancelOutside(true)
                        .setViewListener(v1 -> {
                        })
                        .show();
            }
        });
device.gif

是彈出來了但是還沒有關(guān)聯(lián)上磷雇,所以還要定義一個(gè)狀態(tài)回調(diào)接口來獲取每次操作狀態(tài)回調(diào)
關(guān)鍵代碼

    public IBehaviorChanged getBehaviorChanged() {
        return mBehaviorChanged;
    }

    public void setBehaviorChanged(IBehaviorChanged behaviorChanged) {
        mBehaviorChanged = behaviorChanged;
    }

    public interface IBehaviorChanged {
        void changedState(View bottomSheet, int state);

        void changedOffset(View bottomSheet, float slideOffset);
    }
            BottomSheetBehavior behavior = BottomSheetBehavior.from(bottomSheet);
            // 初始為展開狀態(tài)
            behavior.setState(BottomSheetBehavior.STATE_EXPANDED);
            behavior.setPeekHeight(0);
            //注意這句,初始的時(shí)候給一個(gè)默認(rèn)值躏救,因?yàn)槟J(rèn)是展開狀態(tài)唯笙,所以給的狀態(tài)是STATE_EXPANDED,具體可以根據(jù)需求來
            if (mBehaviorChanged != null)
                mBehaviorChanged.changedState(null, BottomSheetBehavior.STATE_EXPANDED);
            behavior.setBottomSheetCallback(new BottomSheetBehavior.BottomSheetCallback() {
                @Override
                public void onStateChanged(@NonNull View bottomSheet, int newState) {
                    //這里加上這句話是因?yàn)樾枰褷顟B(tài)加上盒使,不然的話滑出屏幕會(huì)有黑色陰影
                    //具體可以看源碼
                    if (newState == BottomSheetBehavior.STATE_HIDDEN || newState == BottomSheetBehavior.STATE_COLLAPSED) {
                        dismiss();
                    }
                    if (mBehaviorChanged != null)
                        mBehaviorChanged.changedState(bottomSheet, newState);
                }

                @Override
                public void onSlide(@NonNull View bottomSheet, float slideOffset) {
                    if (mBehaviorChanged != null)
                        mBehaviorChanged.changedOffset(bottomSheet, slideOffset);
                }
            });
  因?yàn)槭謾C(jī)坐標(biāo)系統(tǒng)默認(rèn)是從屏幕左上角開始計(jì)算
  我在這里默認(rèn)給了dialog的高度是1280/(3/2)崩掘,也就是用1280-1280/(3/2)就是視頻的最小高度
  得到高度之后還要繼續(xù)計(jì)算高度占比來進(jìn)行等比例縮放:
  Dialog 高度占比 = 1280/(3/2)/1280≈0.67
  VideoView 高度占比 = 1280-(1280/(3/2))/1280≈0.33
  所以0.33這個(gè)系數(shù)就是setScaleX()和setScaleY()的縮放比
  因?yàn)镈ialog滑動(dòng)距離一直在改變,得到滑動(dòng)距離之后用當(dāng)前距離除以總高度少办,就是縮放比
  當(dāng)Dialog滑動(dòng)到屏幕最大之后也就會(huì)變成1280/1280=1.0f苞慢,所以視頻也要恢復(fù)初始大小scale(1.0f)
  可以看到視頻始終是在屏幕正上方,中心點(diǎn)在屏幕寬度的1/2處英妓,進(jìn)行縮放,所以當(dāng)Dialog彈出的時(shí)候videoView的x,y坐標(biāo)可以固定在這個(gè)地方挽放,初始狀態(tài)也就是從0開始
        if (scale) {
            float width = Screen.getWidth();
            float x = width / 2f;
            videoView.setPivotX(x);
            videoView.setPivotY(0);
        } else {
            videoView.setPivotX(0);
            videoView.setPivotY(0);
        }

這里畫個(gè)圖:


圖.png

按照這個(gè)思路绍赛,查看BottomSheetCallback回調(diào),這里來添加一個(gè)Log打印一下bottomSheet的Y坐標(biāo)

           behavior.setBottomSheetCallback(new BottomSheetBehavior.BottomSheetCallback() {
                @Override
                public void onStateChanged(@NonNull View bottomSheet, int newState) {
                }

                @Override
                public void onSlide(@NonNull View bottomSheet, float slideOffset) {
                    Log.e("BottomSheetCallback","bottomSheet"+bottomSheet.getY());
            });

            com.behavior.bottombehaviormaster E/BottomSheetCallback: bottomSheet382.0
             ~
             ~
             ~
            com.behavior.bottombehaviormaster E/BottomSheetCallback: bottomSheet1230.0

因太長(zhǎng)這里取最大值和最小值來比較一下1230~382 之間的高度差是848辑畦,為什么是1230呢吗蚌?因?yàn)闆]有計(jì)算系統(tǒng)狀態(tài)欄的高度,默認(rèn)是50dp航闺,所以要加上這50dp


接下來褪测,實(shí)現(xiàn)過程在回調(diào)方法中添加按比例縮放代碼及x,y坐標(biāo)代碼

                bcs.setBehaviorChanged(new BaseBottomSheetDialog.IBehaviorChanged() {
                    @Override
                    public void changedState(View bottomSheet, int state) {
                        if (state == BottomSheetBehavior.STATE_EXPANDED) {
                                float width = Screen.getWidth();
                                float height = Screen.getHeight();
                                float x = width / 2f;
                                float scale = height - view.getHeight();
                                videoView.setScaleX(scale / height);
                                videoView.setScaleY(scale / height);
                                videoView.setPivotX(x);
                                videoView.setPivotY(0);
                            });
                        }
                    }

                    @Override
                    public void changedOffset(View bottomSheet, float slideOffset) {
                        startAnimator(bottomSheet);
                    }
                });
  /**
  *  根據(jù)滑動(dòng)高度進(jìn)行縮放
  * @param parent 
  */
  private void startAnimator(View parent) {
        float width = Screen.getWidth();
        float height = Screen.getHeight();
        float x = width / 2f;
        float py = (parent.getY() + 50) / height;
        videoView.setScaleX(py);
        videoView.setScaleY(py);
        videoView.setPivotX(x);
        videoView.setPivotY(0);
    }

接下來看實(shí)現(xiàn)結(jié)果:
有一些需要優(yōu)化的地方,比如彈出Dialog不顯示StatusBar潦刃,彈出Dialog背景會(huì)變暗侮措,解決掉這兩個(gè)問題之后基本符合。

device.gif

device.gif

附項(xiàng)目地址:https://github.com/futureLix/behavior

第一次發(fā)乖杠,不太會(huì)排版分扎,見諒!

希望疫情早日消散胧洒,武漢加油畏吓!中國加油!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末卫漫,一起剝皮案震驚了整個(gè)濱河市菲饼,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌列赎,老刑警劉巖宏悦,帶你破解...
    沈念sama閱讀 206,378評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異包吝,居然都是意外死亡饼煞,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門诗越,熙熙樓的掌柜王于貴愁眉苦臉地迎上來砖瞧,“玉大人,你說我怎么就攤上這事嚷狞】榇伲” “怎么了?”我有些...
    開封第一講書人閱讀 152,702評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵感耙,是天一觀的道長(zhǎng)褂乍。 經(jīng)常有香客問我,道長(zhǎng)即硼,這世上最難降的妖魔是什么逃片? 我笑而不...
    開封第一講書人閱讀 55,259評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上褥实,老公的妹妹穿的比我還像新娘呀狼。我一直安慰自己,他們只是感情好损离,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,263評(píng)論 5 371
  • 文/花漫 我一把揭開白布哥艇。 她就那樣靜靜地躺著,像睡著了一般僻澎。 火紅的嫁衣襯著肌膚如雪貌踏。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,036評(píng)論 1 285
  • 那天窟勃,我揣著相機(jī)與錄音祖乳,去河邊找鬼。 笑死秉氧,一個(gè)胖子當(dāng)著我的面吹牛眷昆,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播汁咏,決...
    沈念sama閱讀 38,349評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼亚斋,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了攘滩?” 一聲冷哼從身側(cè)響起帅刊,我...
    開封第一講書人閱讀 36,979評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎漂问,沒想到半個(gè)月后厚掷,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡级解,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,938評(píng)論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了田绑。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片勤哗。...
    茶點(diǎn)故事閱讀 38,059評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖掩驱,靈堂內(nèi)的尸體忽然破棺而出芒划,到底是詐尸還是另有隱情,我是刑警寧澤欧穴,帶...
    沈念sama閱讀 33,703評(píng)論 4 323
  • 正文 年R本政府宣布民逼,位于F島的核電站,受9級(jí)特大地震影響涮帘,放射性物質(zhì)發(fā)生泄漏拼苍。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,257評(píng)論 3 307
  • 文/蒙蒙 一调缨、第九天 我趴在偏房一處隱蔽的房頂上張望疮鲫。 院中可真熱鬧吆你,春花似錦、人聲如沸俊犯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽燕侠。三九已至者祖,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間绢彤,已是汗流浹背七问。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留杖虾,地道東北人烂瘫。 一個(gè)月前我還...
    沈念sama閱讀 45,501評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像奇适,于是被迫代替她去往敵國和親坟比。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,792評(píng)論 2 345

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

  • /** * 創(chuàng)建者 yf * 創(chuàng)建時(shí)間 2018/8/17 11:51 * 描述 ${TODO} */...
    木葉紛飛閱讀 3,272評(píng)論 0 0
  • Animation Animation類是所有動(dòng)畫(scale嚷往、alpha葛账、translate、rotate)的基...
    四月一號(hào)閱讀 1,900評(píng)論 0 10
  • 1 背景 不能只分析源碼呀皮仁,分析的同時(shí)也要整理歸納基礎(chǔ)知識(shí)籍琳,剛好有人微博私信讓全面說說Android的動(dòng)畫,所以今...
    未聞椛洺閱讀 2,691評(píng)論 0 10
  • 8. Setting Colors Since release v1.4.0, the ColorTemplate...
    ngugg閱讀 674評(píng)論 0 0
  • 手勢(shì)圖片控件 PinchImageView 點(diǎn)擊圖片框架 photoView packagecom.example...
    Ztufu閱讀 717評(píng)論 0 1