实现抖音评论弹窗视频联动缩放效果

有一年多没打开抖音了,因最近疫情影响只能在家里待着,闲来无事刷刷抖音,发现抖音从最开始的弹窗已换成了弹出弹窗视频在最上方并且按一定比例缩放,如下图

image

要实现上图联动效果,接下来提几个方法:

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倍
//设置锚点的X坐标值,以像素为单位。默认是View的中心。
setPivotX(float pivotX)
//设置锚点的Y坐标值,以像素为单位。默认是View的中心。
setPivotX(float pivotX)

其他的我就不一一介绍了,本文主要用到的就是这几个方法,以下是几个方法的介绍:

【Android开发】View的平移、缩放、旋转以及位置、坐标系_eieihihi的专栏-CSDN博客


奔入主题。
接下来讲一下我的思路。
我在这里并没有用特别复杂的功能,所以本文主要是为了实现滑动缩放功能

首先分析一下布局方式,主页一个视频View,然后点击评论按钮弹出底部操作栏。
好,放出布局,一个视频一个按钮

<?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添加点击事件,弹出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

是弹出来了但是还没有关联上,所以还要定义一个状态回调接口来获取每次操作状态回调
关键代码

    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);
            // 初始为展开状态
            behavior.setState(BottomSheetBehavior.STATE_EXPANDED);
            behavior.setPeekHeight(0);
            //注意这句,初始的时候给一个默认值,因为默认是展开状态,所以给的状态是STATE_EXPANDED,具体可以根据需求来
            if (mBehaviorChanged != null)
                mBehaviorChanged.changedState(null, BottomSheetBehavior.STATE_EXPANDED);
            behavior.setBottomSheetCallback(new BottomSheetBehavior.BottomSheetCallback() {
                @Override
                public void onStateChanged(@NonNull View bottomSheet, int newState) {
                    //这里加上这句话是因为需要把状态加上,不然的话滑出屏幕会有黑色阴影
                    //具体可以看源码
                    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);
                }
            });
  因为手机坐标系统默认是从屏幕左上角开始计算
  我在这里默认给了dialog的高度是1280/(3/2),也就是用1280-1280/(3/2)就是视频的最小高度
  得到高度之后还要继续计算高度占比来进行等比例缩放:
  Dialog 高度占比 = 1280/(3/2)/1280≈0.67
  VideoView 高度占比 = 1280-(1280/(3/2))/1280≈0.33
  所以0.33这个系数就是setScaleX()和setScaleY()的缩放比
  因为Dialog滑动距离一直在改变,得到滑动距离之后用当前距离除以总高度,就是缩放比
  当Dialog滑动到屏幕最大之后也就会变成1280/1280=1.0f,所以视频也要恢复初始大小scale(1.0f)
  可以看到视频始终是在屏幕正上方,中心点在屏幕宽度的1/2处,进行缩放,所以当Dialog弹出的时候videoView的x,y坐标可以固定在这个地方,初始状态也就是从0开始
        if (scale) {
            float width = Screen.getWidth();
            float x = width / 2f;
            videoView.setPivotX(x);
            videoView.setPivotY(0);
        } else {
            videoView.setPivotX(0);
            videoView.setPivotY(0);
        }

这里画个图:


图.png

按照这个思路,查看BottomSheetCallback回调,这里来添加一个Log打印一下bottomSheet的Y坐标

           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

因太长这里取最大值和最小值来比较一下1230~382 之间的高度差是848,为什么是1230呢?因为没有计算系统状态栏的高度,默认是50dp,所以要加上这50dp


接下来,实现过程在回调方法中添加按比例缩放代码及x,y坐标代码

                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);
                    }
                });
  /**
  *  根据滑动高度进行缩放
  * @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);
    }

接下来看实现结果:
有一些需要优化的地方,比如弹出Dialog不显示StatusBar,弹出Dialog背景会变暗,解决掉这两个问题之后基本符合。

device.gif

device.gif

附项目地址:https://github.com/futureLix/behavior

第一次发,不太会排版,见谅!

希望疫情早日消散,武汉加油!中国加油!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • /** * 创建者 yf * 创建时间 2018/8/17 11:51 * 描述 ${TODO} */...
    木叶纷飞阅读 3,279评论 0 0
  • Animation Animation类是所有动画(scale、alpha、translate、rotate)的基...
    四月一号阅读 1,915评论 0 10
  • 1 背景 不能只分析源码呀,分析的同时也要整理归纳基础知识,刚好有人微博私信让全面说说Android的动画,所以今...
    未聞椛洺阅读 2,701评论 0 10
  • 8. Setting Colors Since release v1.4.0, the ColorTemplate...
    ngugg阅读 704评论 0 0
  • 手势图片控件 PinchImageView 点击图片框架 photoView packagecom.example...
    Ztufu阅读 721评论 0 1