当前位置:   article > 正文

android视频适配与裁剪_android videoview视频裁剪

android videoview视频裁剪

https://my.oschina.net/u/1403288/blog/634051

摘要: android视频播放时对不同分辨率的手机的视频进行等比例缩放,然后裁剪掉一个方向上的多余部分,使之在不同手机上人的感观是不变的.

首先说下基本背景, 当我们使用android系统原生的VideoView播放视频时, 在XML中给它设置的一个尺寸, 但最终视频开始播放后, VideoView实际的尺寸可能并不是这个尺寸设置的大小. VideoView在测量自身的尺寸时会依据视频的真实尺寸来调整自己的大小, 遵循以下规则:

1. 实际视频在VideoView上播放时所有部分都是可见的,或缩小或放大, 总之一定要全部显示出来,不会裁剪实际视频.

2. 尽量保持实际视频的长宽比例, 具体是首先以我们用户定义的长度为标准, 等比例缩放视频大小, 直到长度达到我们定义的长度, 然后宽度(等比例缩放后的宽度)与我们定义的宽比较, 大于则以我们定义的宽度为准, 这样视频会在竖直方向上压缩, 最终播放时也就不会成比例了; 小于则它以视频缩放后的宽度为准, 这样它会比我们定义的高度小,最终播放的效果是等比例的.

最近开发有如下需求:

视频等比例放大,直至一边铺满VideoView(或屏幕)的某一边,另一边超出View的另一边,再移动到View的正中央,这样长边两边会被裁剪掉同样大小的区域,视频看起来不会变形,也即是:先把视频区(实际的大小显示区)与View(定义的大小)区的两个中心点重合, 然后等比例放大或缩小视频区,直至一条边与View的一条边相等,另一条边超过View的另一条边,这时再裁剪掉超出的边, 使视频区与View区大小一样. 这样在不同尺寸的手机上,视频看起来不会变形,只是水平或竖直方向的两端被裁剪了一些.

以上需求用系统的VideoView是无法达到的,所以要自定义VideoView.

最终我们扩展TextureView来自定义一个TextureVideoView, 并应用Matrix进行缩放, 也即是要求出变换的Matrix, 主要按以下步骤来即可求出变换矩阵, 这里直接贴出代码:


  1. //需求:视频等比例放大,直至一边铺满View的某一边,另一边超出View的另一边,再移动到View的正中央,这样长边两边会被裁剪掉同样大小的区域,视频看起来不会变形
  2. //也即是:先把视频区(实际的大小显示区)与View(定义的大小)区的两个中心点重合, 然后等比例放大或缩小视频区,直至一条边与View的一条边相等,另一条边超过
  3. //View的另一条边,这时再裁剪掉超出的边, 使视频区与View区大小一样. 这样在不同尺寸的手机上,视频看起来不会变形,只是水平或竖直方向的两端被裁剪了一些.
  4. private void transformVideo(int videoWidth, int videoHeight) {
  5. if (getHeight() == 0 || getWidth() == 0) {
  6. Log.d(TAG, "transformVideo, getHeight=" + getHeight() + "," + "getWidth=" + getWidth());
  7. return;
  8. }
  9. float sx = (float) getWidth() / (float) videoWidth;
  10. float sy = (float) getHeight() / (float) videoHeight;
  11. Log.d(TAG, "transformVideo, sx=" + sx);
  12. Log.d(TAG, "transformVideo, sy=" + sy);
  13. float maxScale = Math.max(sx, sy);
  14. if (this.matrix == null) {
  15. matrix = new Matrix();
  16. } else {
  17. matrix.reset();
  18. }
  19. //第2步:把视频区移动到View区,使两者中心点重合.
  20. matrix.preTranslate((getWidth() - videoWidth) / 2, (getHeight() - videoHeight) / 2);
  21. //第1步:因为默认视频是fitXY的形式显示的,所以首先要缩放还原回来.
  22. matrix.preScale(videoWidth / (float) getWidth(), videoHeight / (float) getHeight());
  23. //第3步,等比例放大或缩小,直到视频区的一边超过View一边, 另一边与View的另一边相等. 因为超过的部分超出了View的范围,所以是不会显示的,相当于裁剪了.
  24. matrix.postScale(maxScale, maxScale, getWidth() / 2, getResizedHeight() / 2);//后两个参数坐标是以整个View的坐标系以参考的
  25. Log.d(TAG, "transformVideo, maxScale=" + maxScale);
  26. setTransform(matrix);
  27. postInvalidate();
  28. Log.d(TAG, "transformVideo, videoWidth=" + videoWidth + "," + "videoHeight=" + videoHeight);
  29. }

然后在OnVideoSizeChangedListener里面监听获取到了视频的实际尺寸后调用以上方法即可.

完整项目代码: https://github.com/linsea/MatrixScale


声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Li_阴宅/article/detail/970121
推荐阅读
相关标签
  

闽ICP备14008679号