jjzjj

Android检测来自URL的视频是否为纵向

coder 2023-12-14 原文

在我的 Android 应用程序中,我有一个自定义视频播放器,可以播放来自 URL 的视频。但是,当视频以纵向拍摄时,视频会在应用程序中逆时针旋转 90 度。

有没有办法让我获取视频的方向并相应地旋转 View ?我需要动态检测方向,因为该应用仍应能够播放横向视频(它目前没有方向问题)。

这是我用来显示视频的内容:

// create the video controller
videoController = new MediaController(this);
videoControllerExists = true;
videoController.setAnchorView(mediaHolder);
videoController.setPadding(0, 0, 0, bottomButtonHeight);

s3Video = new VideoView(MediaPreview.this);
s3Video.setVideoURI(Uri.parse(mediaURL));
s3Video.setMediaController(videoController);
// do something here to determine if video is in portrait orientation
DisplayMetrics displaymetrics = new DisplayMetrics();
getWindowManager().getDefaultDisplay().getMetrics(displaymetrics);
int h = displaymetrics.heightPixels;
int w = displaymetrics.widthPixels;

// add the video player to the view and play
mediaHolder.addView(s3Video);
s3Video.start();
s3Video.bringToFront();
videoController.requestFocus();

如果您正在尝试,可以使用下面的肖像视频进行测试: https://dl.dropboxusercontent.com/u/1000620/portrait.mp4

编辑

看了一些示例后,我从 VideoView 切换到 TextureView 并且能够旋转视频。不过,我仍然不知道如何从 URL 中检测视频的方向,所以所有视频现在都是旋转的。

如何从 URL 检测视频的方向?

这是我更新后的代码:

public class MediaPreview extends SherlockActivity implements YouTubePlayer.OnInitializedListener, TextureView.SurfaceTextureListener, OnBufferingUpdateListener, OnCompletionListener, OnPreparedListener,
OnVideoSizeChangedListener, MediaPlayerControl {

    private MediaPlayer s3VideoPlayer;
    private TextureView s3TextureView;
    private Surface s;

    @Override
    protected void onCreate(Bundle savedInstanceState) {

    // set up textureview
        s3TextureView = (TextureView) findViewById(R.id.s3_video);
        s3TextureView.setSurfaceTextureListener(this);
    }

    @Override
    public void onSurfaceTextureAvailable(SurfaceTexture surface, int width,
            int height) {
        s = new Surface(surface);
        if(isFromVideo){
            playAmazonVideo(mediaHolder);
        }

    }

    private void playAmazonVideo(FrameLayout mediaHolder){

            // set up mediacontroller
            videoController = new MediaController(this);
            videoControllerExists = true;
            videoController.setAnchorView(mediaHolder);
            videoController.setPadding(0, 0, 0, bottomButtonHeight);

            // set scale for textureview
            DisplayMetrics displaymetrics = new DisplayMetrics();
            getWindowManager().getDefaultDisplay().getMetrics(displaymetrics);
            int h = displaymetrics.heightPixels;
            int w = displaymetrics.widthPixels;
            LayoutParams videoLayoutParams = new LayoutParams(h, w);
            videoLayoutParams.gravity = Gravity.CENTER;
            s3TextureView.setScaleX(h/w);
            s3TextureView.setLayoutParams(videoLayoutParams);

       // Setting up video view using Media Player
        try{
            s3VideoPlayer = MediaPlayer.create(this, Uri.parse(mediaURL));
            s3VideoPlayer.setSurface(s);
            s3VideoPlayer.setOnBufferingUpdateListener(this);
            s3VideoPlayer.setOnCompletionListener(this);
            s3VideoPlayer.setOnPreparedListener(this);
            s3VideoPlayer.setScreenOnWhilePlaying(true);
            s3VideoPlayer.setOnVideoSizeChangedListener(this);
            s3VideoPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
            s3VideoPlayer.start();
    }

    @Override
    public void onPrepared(MediaPlayer mp) {
        if(isAmazon && mediaURL.length() > 0){
            mediaProgress.setVisibility(View.GONE);
            videoController.setMediaPlayer(this);
            videoController.setEnabled(true);
            videoController.show();     
        }    
    }

    @Override
    public boolean onTouchEvent(MotionEvent event){
        if(isAmazon){
            Log.d(TAG, "showing video controller");
            videoController.show();    
        }
        return false;
    }

 @Override
    public boolean canPause() {
        return true;
    }

    @Override
    public boolean canSeekBackward() {
        return true;
    }

    @Override
    public boolean canSeekForward() {
        return true;
    }

   @Override
    public int getCurrentPosition() {
        return s3VideoPlayer.getCurrentPosition();
    }

    @Override
    public int getDuration() {
        return s3VideoPlayer.getDuration();        
    }

    @Override
    public boolean isPlaying() {
        return s3VideoPlayer.isPlaying();
    }

 @Override
    public void pause() {
        s3VideoPlayer.pause();
    }

    @Override
    public void seekTo(int pos) {
        s3VideoPlayer.seekTo(pos);

    }

    @Override
    public void start() {
        s3VideoPlayer.start();

    }


    }

编辑 #2

我一直在研究检测视频方向的选项并找到了 http://wseemann.github.io/FFmpegMediaMetadataRetriever/

不幸的是,即使我查看视频文件的 exif,我也可以看到方向值为 90,这似乎并没有使视频旋转。所以如果有人有任何其他解决方案,请分享!

编辑 #3

这是我尝试过的其他东西。我看到您可以为 MediaPlayer 设置 OnVideoSizeChangedListener 以获取视频的宽度和高度。但是对于我使用的视频,宽度总是大于高度,播放器根据旋转 EXIF 值知道是否旋转视频。所以我也不能用这个...

最佳答案

我无法弄清楚旋转问题,所以我最终只是更新了我的 Rails 应用程序以在更新时保存视频的旋转。然后,当我从网站获取视频到 Android 应用程序时,我还检索视频的旋转并相应地更新纹理 View 旋转。这确实不是一个理想的方法,但我找不到任何其他解决方案。

最后是这样的:

        // set scale for textureview
        final Intent i = getIntent();
        videoRotation = i.getIntExtra("videoRotation", 0);
        Log.d(TAG, "setting video rotation...");
        DisplayMetrics displaymetrics = new DisplayMetrics();
        getWindowManager().getDefaultDisplay().getMetrics(displaymetrics);
        int h = displaymetrics.heightPixels;
        int w = displaymetrics.widthPixels;
        Log.d(TAG, "hxw: " + h + w);
        LayoutParams videoLayoutParams = new LayoutParams(h, w);
        videoLayoutParams.gravity = Gravity.CENTER;
        s3TextureView.setScaleX(h/w);    
        s3TextureView.setRotation(Math.abs(360-videoRotation));
        s3TextureView.setLayoutParams(videoLayoutParams);
        Log.d(TAG, "done setting video rotation");

关于Android检测来自URL的视频是否为纵向,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21835119/

有关Android检测来自URL的视频是否为纵向的更多相关文章

  1. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  2. ruby-on-rails - rails : save file from URL and save it to Amazon S3 - 2

    从给定URL下载文件并立即将其上传到AmazonS3的更直接的方法是什么(+将有关文件的一些信息保存到数据库中,例如名称、大小等)?现在,我既不使用Paperclip,也不使用Carrierwave。谢谢 最佳答案 简单明了:require'open-uri'require's3'amazon=S3::Service.new(access_key_id:'KEY',secret_access_key:'KEY')bucket=amazon.buckets.find('image_storage')url='http://www.ex

  3. ruby - 检查数组是否在增加 - 2

    这个问题在这里已经有了答案:Checktoseeifanarrayisalreadysorted?(8个答案)关闭9年前。我只是想知道是否有办法检查数组是否在增加?这是我的解决方案,但我正在寻找更漂亮的方法:n=-1@arr.flatten.each{|e|returnfalseife

  4. ruby - 如何使用 Ruby aws/s3 Gem 生成安全 URL 以从 s3 下载文件 - 2

    我正在编写一个小脚本来定位aws存储桶中的特定文件,并创建一个临时验证的url以发送给同事。(理想情况下,这将创建类似于在控制台上右键单击存储桶中的文件并复制链接地址的结果)。我研究过回形针,它似乎不符合这个标准,但我可能只是不知道它的全部功能。我尝试了以下方法:defauthenticated_url(file_name,bucket)AWS::S3::S3Object.url_for(file_name,bucket,:secure=>true,:expires=>20*60)end产生这种类型的结果:...-1.amazonaws.com/file_path/file.zip.A

  5. ruby - RuntimeError(自动加载常量 Apps 多线程时检测到循环依赖 - 2

    我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("

  6. ruby - 检查字符串是否包含散列中的任何键并返回它包含的键的值 - 2

    我有一个包含多个键的散列和一个字符串,该字符串不包含散列中的任何键或包含一个键。h={"k1"=>"v1","k2"=>"v2","k3"=>"v3"}s="thisisanexamplestringthatmightoccurwithakeysomewhereinthestringk1(withspecialcharacterslike(^&*$#@!^&&*))"检查s是否包含h中的任何键的最佳方法是什么,如果包含,则返回它包含的键的值?例如,对于上面的h和s的例子,输出应该是v1。编辑:只有字符串是用户定义的。哈希将始终相同。 最佳答案

  7. ruby-on-rails - Ruby 检查日期时间是否为 iso8601 并保存 - 2

    我需要检查DateTime是否采用有效的ISO8601格式。喜欢:#iso8601?我检查了ruby​​是否有特定方法,但没有找到。目前我正在使用date.iso8601==date来检查这个。有什么好的方法吗?编辑解释我的环境,并改变问题的范围。因此,我的项目将使用jsapiFullCalendar,这就是我需要iso8601字符串格式的原因。我想知道更好或正确的方法是什么,以正确的格式将日期保存在数据库中,或者让ActiveRecord完成它们的工作并在我需要时间信息时对其进行操作。 最佳答案 我不太明白你的问题。我假设您想检查

  8. ruby - 检查日期是否在过去 7 天内 - 2

    我的日期格式如下:"%d-%m-%Y"(例如,今天的日期为07-09-2015),我想看看是不是在过去的七天内。谁能推荐一种方法? 最佳答案 你可以这样做:require"date"Date.today-7 关于ruby-检查日期是否在过去7天内,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/32438063/

  9. ruby - 如何验证 IO.copy_stream 是否成功 - 2

    这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

  10. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

随机推荐