我要做的是制作一个可以在屏幕关闭时录制视频或音频的应用。我们要么开始录制然后关闭屏幕,要么录音机应该用语音或任何其他外部命令触发。在任何一种情况下,应用程序都应继续录制。
请提供您的代码。
答案 0 :(得分:14)
大多数情况下,当某些事情需要离线完成时,它会在service
中完成。
假设您不知道服务是什么,请参阅文档:
服务是表示一个的应用程序组件 应用程序希望执行较长时间运行而不是 与用户交互或为其他人提供功能 要使用的应用程序每个服务类必须具有相应的 声明在其包的AndroidManifest.xml中。服务 可以使用Context.startService()和Context.bindService()启动。
所以基本上,它只是在后台运行,包括屏幕关闭时。现在,您需要做的就是将您的录制代码放入服务中。您可能会遇到系统杀死您的服务的问题,如android文档指定的那样:
请注意,这意味着您的服务大部分时间都在运行 如果它受到严重的记忆压力,就会被系统杀死。如果这 发生这种情况,系统稍后会尝试重启服务。一个 这样做的重要后果是,如果你实施 onStartCommand()用于安排异步或异步完成的工作 另一个线程,那么你可能想要使用START_FLAG_REDELIVERY 系统会为您重新提供一个Intent,以便它不会丢失 如果您的服务在处理过程中被终止。
在这种情况下,您需要添加某些代码以使服务保持在前景中,并确保它不会被杀死:
startForeground();//keeps service from getting destroyed.
这可能提供的唯一问题是持续通知服务处于活动状态的时间。
您也可以使用startSticky()
:
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
// We want this service to continue running until it is explicitly
// stopped, so return sticky.
return START_STICKY;
}
现在,说你已完成录音。您可以使用:
stopSelf();
停止服务,或者:
Context.stopService();
但是这个通知并不是什么大问题,因为在服务运行时屏幕很可能已关闭。您还可以自定义通知,用户最好知道后台发生了什么。
这里有一些示例录制代码:
public class VideoCapture extends Activity implements OnClickListener, SurfaceHolder.Callback {
MediaRecorder recorder;
SurfaceHolder holder;
boolean recording = false;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
recorder = new MediaRecorder();
initRecorder();
setContentView(R.layout.main);
SurfaceView cameraView = (SurfaceView) findViewById(R.id.CameraView);
holder = cameraView.getHolder();
holder.addCallback(this);
holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
cameraView.setClickable(true);
cameraView.setOnClickListener(this);
}
private void initRecorder() {
recorder.setAudioSource(MediaRecorder.AudioSource.DEFAULT);
recorder.setVideoSource(MediaRecorder.VideoSource.DEFAULT);
CamcorderProfile cpHigh = CamcorderProfile
.get(CamcorderProfile.QUALITY_HIGH);
recorder.setProfile(cpHigh);
recorder.setOutputFile("/sdcard/videocapture_example.mp4");
recorder.setMaxDuration(50000); // 50 seconds
recorder.setMaxFileSize(5000000); // Approximately 5 megabytes
}
private void prepareRecorder() {
recorder.setPreviewDisplay(holder.getSurface());
try {
recorder.prepare();
} catch (IllegalStateException e) {
e.printStackTrace();
finish();
} catch (IOException e) {
e.printStackTrace();
finish();
}
}
public void onClick(View v) {
if (recording) {
recorder.stop();
recording = false;
// Let's initRecorder so we can record again
initRecorder();
prepareRecorder();
} else {
recording = true;
recorder.start();
}
}
public void surfaceCreated(SurfaceHolder holder) {
prepareRecorder();
}
public void surfaceChanged(SurfaceHolder holder, int format, int width,
int height) {
}
public void surfaceDestroyed(SurfaceHolder holder) {
if (recording) {
recorder.stop();
recording = false;
}
recorder.release();
finish();
}
}
(How can I capture a video recording on Android?)
再次,将该代码放入长时间运行的前台服务中,如前所述。
如果您需要更多帮助,请随时提出。
〜Ruchir