Подтвердить что ты не робот

Android-камера записывает видео, но играет вверх ногами

Я записываю видео, используя приведенный ниже код, и он отлично записывается, но когда он воспроизводит видео, он воспроизводит его вверх дном.

Я пробовал настройки mrec.setOrientationHint(180) до mrec.prepare();, но это было бесполезно. Любые подсказки?

import java.io.File;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import android.app.Activity;
import android.hardware.Camera;
import android.media.MediaRecorder;
import android.os.Bundle;
import android.os.Environment;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.view.Surface;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.Window;

/**
 * @author SANA HASSAN
 */
public class CameraSurfaceView extends Activity {

    private Preview mPreview;
    private MediaRecorder mrec = new MediaRecorder();
    private int cameraId = 0;
    private Camera mCamera;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        requestWindowFeature(Window.FEATURE_NO_TITLE);

        mPreview = new Preview(this);
        setContentView(mPreview);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        menu.add(0, 0, 0, "Start");
        menu.add(0, 1, 0, "Stop");
        return super.onCreateOptionsMenu(menu);
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()) {
            case 0:
                try {
                    startRecording();
                } 
                catch (Exception e) {
                    e.printStackTrace();
                    mrec.release();
                }
                break;

            case 1: 
                mrec.stop();
                mrec.release();
                mrec = null;
                break;

            default:
                break;
        }
        return super.onOptionsItemSelected(item);
    }

    protected void startRecording() throws IOException  {

        mrec = new MediaRecorder();
        mrec.setCamera(mCamera);
        mCamera.unlock();
        File directory = new File(Environment.getExternalStorageDirectory()+"/NICUVideos");
        directory.mkdirs();
        mrec.setAudioSource( MediaRecorder.AudioSource.MIC);
        mrec.setVideoSource(MediaRecorder.VideoSource.CAMERA);
        mrec.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);
        mrec.setOutputFile(Environment.getExternalStorageDirectory()+"/NICUVideos/"+System.currentTimeMillis()+".mp4"); 
        mrec.setPreviewDisplay(mPreview.getHolder().getSurface());
        mrec.setVideoSize(640, 480);

        Method[] methods = mrec.getClass().getMethods();
        for (Method method: methods){
            try{
                if(method.getName().equals("setAudioEncodingBitRate")){
                    method.invoke(mrec, 12200);
                }
                else if(method.getName().equals("setVideoEncodingBitRate")){
                    method.invoke(mrec, 800000);
                }
                else if(method.getName().equals("setAudioSamplingRate")){
                    method.invoke(mrec, 8000);
                }
                else if(method.getName().equals("setVideoFrameRate")){
                    method.invoke(mrec, 20);
                }
            }
            catch (IllegalArgumentException e) {
                e.printStackTrace();
            } 
            catch (IllegalAccessException e) {
                e.printStackTrace();
            } 
            catch (InvocationTargetException e) {
                e.printStackTrace();
            }
        }
        mrec.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
        mrec.setVideoEncoder(MediaRecorder.VideoEncoder.MPEG_4_SP);
        mrec.setMaxDuration(60000); // 60 seconds
        mrec.setMaxFileSize(10000000); // Approximately 10 megabytes
        mrec.prepare();
        mrec.start();
    }

    protected void stopRecording() {
        mrec.stop();
        mrec.release();
        mCamera.release();
    }

    class Preview extends SurfaceView implements SurfaceHolder.Callback {
        SurfaceHolder mHolder;
        Activity activity;

        Preview(Activity activity) {
            super(activity);
            mHolder = getHolder();
            mHolder.addCallback(this);
            mHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
        }

        public void surfaceCreated(SurfaceHolder holder) {

            Camera.CameraInfo info=new Camera.CameraInfo();
            for (int i=0; i < Camera.getNumberOfCameras(); i++) {
                Camera.getCameraInfo(i, info);
                if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
                    mCamera=Camera.open(i);
                    cameraId = i;
                }
            }

            try {
               mCamera.setPreviewDisplay(holder);
            } catch (IOException exception) {
                mCamera.release();
                mCamera = null;
            }
        }

        public void surfaceDestroyed(SurfaceHolder holder) {
            mCamera.stopPreview();
            mCamera.release();
            mCamera = null;
        }

        public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {
            setCameraDisplayOrientation(mCamera);
            mCamera.startPreview();
        }

        public void setCameraDisplayOrientation(Camera camera) {
            Camera.CameraInfo info = new Camera.CameraInfo();
            Camera.getCameraInfo(cameraId, info);

            int rotation = CameraSurfaceView.this.getWindowManager().getDefaultDisplay().getRotation();
            int degrees = 0;
            switch (rotation) {
                case Surface.ROTATION_0: degrees = 0; break;
                case Surface.ROTATION_90: degrees = 90; break;
                case Surface.ROTATION_180: degrees = 180; break;
                case Surface.ROTATION_270: degrees = 270; break;
            }

            int result;
            if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
                result = (info.orientation + degrees) % 360;
                result = (360 - result) % 360;  // compensate the mirror
            } else {  // back-facing
                result = (info.orientation - degrees + 360) % 360;
            }
            Log.d(Vars.TAG, "Result = "+result);
            camera.setDisplayOrientation(result);
        }
    }

}
4b9b3361

Ответ 1

Эта проблема связана с вращением обработки Android, просто устанавливая некоторые метаданные вместо фактического вращения видео, а также некоторое программное обеспечение для воспроизведения, а затем игнорируя эту настройку.

Как отмечено в в документах:

Обратите внимание, что некоторые видеоплееры могут игнорировать матрицу компоновки во время воспроизведения.

Ваши варианты - либо использовать другое программное обеспечение воспроизведения, которое понимает установленные метаданные, либо перекодировать видео после его записи в правильную ориентацию. Из вашего описания не ясно, какое из них было бы лучшим решением в вашем случае.

Ответ 2

Это нужно вызывать перед методом mrec.prepare();

setOrientationHint(degrees);

Ссылка

EDIT: Попробуйте mCamera.setDisplayOrientation(degrees);

0 for landscape
90 for portrait
180 & 270 don't work very well and give weird results.

Некоторые старые игроки и кодировщики не интерпретируют этот флаг, поэтому видео воспроизводится с ног на голову.

Ответ 3

У меня такая же проблема, я отмечаю, что угол ориентации предварительного просмотра камеры и угол видеозаписи не совпадают. Поэтому я использую этот метод для ориентации изменения при записи видео:

public static int getVideoOrientationAngle(Activity activity, int cameraId) { //The param cameraId is the number of the camera.
    int angle;
    Display display = activity.getWindowManager().getDefaultDisplay();
    int degrees = display.getRotation();
    android.hardware.Camera.CameraInfo info =
            new android.hardware.Camera.CameraInfo();
    android.hardware.Camera.getCameraInfo(cameraId, info);
    switch (degrees) {
        case Surface.ROTATION_0: 
            angle = 90; 
            break;
        case Surface.ROTATION_90:
            angle = 0;
            break;
        case Surface.ROTATION_180:
            angle = 270;
            break;
        case Surface.ROTATION_270:
            angle = 180;
            break;
        default:
            angle = 90;
            break;
    }
    if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT)
        angle = (angle + 180) % 360;

    return angle;
}

И для ориентации ориентации на предварительный просмотр камеры:

 public static int setCameraDisplayOrientation(Activity activity,
                                              int cameraId, android.hardware.Camera camera) {
    android.hardware.Camera.CameraInfo info =
            new android.hardware.Camera.CameraInfo();
    android.hardware.Camera.getCameraInfo(cameraId, info);
    int rotation = activity.getWindowManager().getDefaultDisplay()
            .getRotation();
    int degrees = 0;
    switch (rotation) {
        case Surface.ROTATION_0:
            degrees = 0;
            break;
        case Surface.ROTATION_90:
            degrees = 90;
            break;
        case Surface.ROTATION_180:
            degrees = 180;
            break;
        case Surface.ROTATION_270:
            degrees = 270;
            break;
    }

    int result;
    if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
        result = (info.orientation + degrees) % 360;
        result = (360 - result) % 360;  // compensate the mirror
    } else {  // back-facing
        result = (info.orientation - degrees + 360) % 360;
    }
    camera.setDisplayOrientation(result);
    return result;
}

Обратите внимание, что важно знать, находится ли камера спереди или сзади.

Ответ 4

Я знаю вашу проблему,

Использование видео Media Recorder от Camera, поэтому вам нужно rotate Media Recorder. использование ниже кода должно быть исправлено вашей проблемой.

/**
 * 
 * @param mMediaRecorder
 * @return
 */
public static MediaRecorder rotateBackVideo(MediaRecorder mMediaRecorder) {
    /**
     * Define Orientation of video in here,
     * if in portrait mode, use value = 90,
     * if in landscape mode, use value = 0
     */
    switch (CustomCamera.current_orientation) {
        case 0:
            mMediaRecorder.setOrientationHint(90);
            break;
        case 90:
            mMediaRecorder.setOrientationHint(180);
            break;
        case 180:
            mMediaRecorder.setOrientationHint(270);
            break;
        case 270:
            mMediaRecorder.setOrientationHint(0);
            break;
    }

    return mMediaRecorder;
}

Должен добавить метод prepare():

// Step 5: Set the preview output
    /**
     * Define Orientation of image in here,
     * if in portrait mode, use value = 90,
     * if in landscape mode, use value = 0
     */
    CustomCamera.mMediaRecorder = Utils.rotateBackVideo(CustomCamera.mMediaRecorder);
    CustomCamera.mMediaRecorder.setPreviewDisplay(mCameraPreview.getHolder().getSurface());

Спасибо

Ответ 5

Используйте OrientationEventListener и отслеживайте значение поворота, когда устройство повернуто code здесь. Этот код применит поворот к камере, но для записи вам нужно применить поворот к MediaRecorder. Когда вы начинаете запись только mMediaRecorder.setOrientationHint(rotation) до mMediaRecorder.prepare(). Это решает мою проблему.

Ответ 6

Наконец, я обнаружил, что телефоны Motorola имеют проблемы с воспроизведением видео, записанного в портретном режиме.

Чтобы преодолеть поворот видео, лучшим решением, которое я принял, было загрузка видео на сервер и запуск ffmpeg при использовании команды ffmpeg -i input.mp4 -c:v mpeg4 -c:a copy -c:s copy -vf "transpose=2" output.mp4

Если вы чувствуете, что есть другой способ, пожалуйста, дайте мне знать.

Ответ 7

еще одно решение - повернуть вашу деятельность так, чтобы ее ориентация была такой же, как ориентация датчика. То есть, пейзаж для задней камеры и перевернутый портрет для передней камеры. Этот бит не будет фиксировать зеркальный эффект для передней камеры. Еще одна трудность заключается в том, что вам придется реализовать свой интерфейс в этих повернутых действиях, чтобы следить за ориентацией деятельности.

Ответ 8

вращайте свой MediaRecorder, как показано ниже, в зависимости от степени, которую вы использовали в ориентации дисплея камеры для видеозаписи с фронтальной камерой.

Display display = ((WindowManager)getContext().getSystemService(WINDOW_SERVICE)).getDefaultDisplay();
        parameters.setPreviewSize(height, width);
        if(display.getRotation() == Surface.ROTATION_0)
        {

            mCamera.setDisplayOrientation(90);
            mMediaRecorder.setOrientationHint(270);
        }

        if(display.getRotation() == Surface.ROTATION_90)
        {
            mMediaRecorder.setOrientationHint(180);
        }

        if(display.getRotation() == Surface.ROTATION_180)
        {
            mMediaRecorder.setOrientationHint(270);
        }

        if(display.getRotation() == Surface.ROTATION_270)
        {
            mCamera.setDisplayOrientation(180);
            mMediaRecorder.setOrientationHint(0);
        }

Ответ 9

Вот код для пользовательской портретной камеры, установите правильное вращение изображения и видео:


private OrientationEventListener orientationEventListener;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    //...
    orientationEventListener = new OrientationEventListener(this) {
        @Override
        public void onOrientationChanged(int orientation) {
            if (orientation == ORIENTATION_UNKNOWN) return;

            flashButton.setRotation(-(orientation));
            cameraButton.setRotation(-(orientation));

            if (camera != null) {
                Parameters parameters = camera.getParameters();
                CameraInfo info = new CameraInfo();
                Camera.getCameraInfo(selectedCamera, info);
                orientation = (orientation + 45) / 90 * 90;
                int rotation = 0;
                if (info.facing == CameraInfo.CAMERA_FACING_FRONT) {
                    rotation = (info.orientation - orientation + 360) % 360;
                } else {  // back-facing camera
                    rotation = (info.orientation + orientation) % 360;
                }
                parameters.setRotation(rotation);
                if (!isRecording) {
                    mediaRecorder.setOrientationHint(rotation);
                }

                camera.setParameters(parameters);
            }
        }
    };
}


@Override
protected void onResume() {
    super.onResume();
    //...
    orientationEventListener.enable();
}

@Override
protected void onPause() {
    super.onPause();
    orientationEventListener.disable();
    //...
}

Тест с портретной ориентацией. Помните, чтобы поместить его в свой манифест, чтобы проверить код. Я не знаю, работает ли в ландшафте.

<activity android:name=".activities.CameraActivity"
          android:screenOrientation="portrait">

Ответ 10

для портретного режима установите значение mediaRecorder.setOrientationHint(90); степень аналогична ориентации вашей камеры myCamera.setDisplayOrientation(90);