[英]Android app video recording when screen off
我想要做的是制作一个可以在屏幕关闭时录制视频或音频的应用。 我们要么开始录制然后关闭屏幕,要么录音机应该用语音或任何其他外部命令触发。 在任何一种情况下,应用程序都应继续录制。
请提供您的代码。
大多数情况下,当需要离线完成某些操作时,它是在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();
}
}
再次,将该代码放入长时间运行的前台服务中,正如我之前解释的那样。
如果您需要更多帮助,请随时提出。
〜Ruchir
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.