当前位置: 首页 > news >正文

页面模板够30条上海百度seo

页面模板够30条,上海百度seo,vs网站搜索栏怎么做,模具 东莞网站建设在“Android笔记(八):基于CameraX库结合Compose和传统视图组件PreviewView实现照相机画面预览和照相功能”,文中介绍了拍照功能的实现,在本文中将介绍结合JetPack Compose和CameraX实现视频的录制。 新建一个项目 在项…

在“Android笔记(八):基于CameraX库结合Compose和传统视图组件PreviewView实现照相机画面预览和照相功能”,文中介绍了拍照功能的实现,在本文中将介绍结合JetPack Compose和CameraX实现视频的录制。
新建一个项目
在这里插入图片描述
在项目中做如下处理:

一、增视频录制的相关依赖库

在项目的模块对应的build.gradle.kt中增加如下的依赖库:
增加CameraX相关库

  val camerax_version = "1.3.0-alpha04"implementation("androidx.camera:camera-core:$camerax_version")implementation("androidx.camera:camera-camera2:${camerax_version}")implementation("androidx.camera:camera-lifecycle:${camerax_version}")implementation("androidx.camera:camera-video:${camerax_version}")implementation("androidx.camera:camera-view:${camerax_version}" )implementation("androidx.camera:camera-extensions:${camerax_version}")

增加Material3中扩展的图标库

  val icon_extended_version = "1.5.4"implementation("androidx.compose.material:material-icons-extended:$icon_extended_version")

注意:因为在前面展示的Android移动应用中使用的图标可选有限,因此增加“androidx.compose.material:material-icons-extended”图标扩展库,丰富图标的选择。

二、在AndroidManifest.xml权限设置使用照相机

  <uses-feature android:name="android.hardware.camera"  android:required="false" /><uses-permission android:name="android.permission.CAMERA" /><uses-permission android:name="android.permission.RECORD_AUDIO" />

三、CameraX视频捕获架构

视频录制需要做两个方面的处理,需要录制视频流和同时录制音频流。然后对音频流和视频流进行压缩处理,最终写入到磁盘保存起来。在图1展示的是这样工作的示意图。
在这里插入图片描述
图1
LifecycleCameraController
CameraX提供了LifecycleCameraController。LifecycleCameraController提供了CameraX的大部分的特性,它是一个高级的控制器类提供了CameraX的核心特性,用于处理照相机的初始化、创建和配置用例(这里的用例提供了将用例参数映射到相机的可用参数的功能,具体这些用例是指:拍照用例:CameraController.IMAGE_CAPTURE、 图像分析用例:CameraController.IMAGE_ANALYSIS、视频录制用例:CameraController.VIDEO_CAPTURE,…),并将它们绑定到一个生命周期拥有者对象。LifecycleCamera监听设备的Motion Sensor(运动感应器),设置目标的旋转参数。

val cameraController: LifecycleCameraController
= LifecycleCameraController(applicationContext).apply {
setEnabledUseCases(CameraController.VIDEO_CAPTURE) //设置用例
}

CameraSelector
CameraSelector用于选择摄像头,具体选择包括:

使用 CameraSelector.DEFAULT_FRONT_CAMERA 请求默认的前置摄像头。
使用 CameraSelector.DEFAULT_BACK_CAMERA 请求默认的后置摄像头。
使用CameraSelector.Builder.addCameraFilter() 按 CameraCharacteristics 过滤可用设备列表。

Recording
Recording是实际上执行录制视频的对象,在活动录制视频时,提供暂停、恢复或停止录制的控件。如果在录制的过程中发生错误,则会初始化一个VideoRecordEvent.Finalized状态,所有的控制将进入空操作。
//创建Recording对象并启动视频录制
val recording = cameraController.startRecording(…)
在这里:startRecording根据录制得到的视频存储处理方式不同有三种形式:
(1)写入到文件

public Recording startRecording(
@NonNull FileOutputOptions outputOptions, //写入到文件的输出参数
@NonNull AudioConfig audioConfig, //音频的配置
@NonNull Executor executor,//将在其上运行事件侦听器的线程池
@NonNull Consumer listener)//处理视频录制的事件监听器

(2)根据文件描述写入到文件

public Recording startRecording(
@NonNull FileDescriptorOutputOptions outputOptions,//写入到文件描述的输出参数
@NonNull AudioConfig audioConfig,//音频的配置
@NonNull Executor executor,//将在其上运行事件侦听器的线程池
@NonNull Consumer listener)//处理视频录制的事件监听器

(3)写入到媒体库

public Recording startRecording(
@NonNull MediaStoreOutputOptions outputOptions,//写入到媒体库的输出参数
@NonNull AudioConfig audioConfig,//音频的配置
@NonNull Executor executor,//将在其上运行事件侦听器的线程池
@NonNull Consumer listener)//处理视频录制的事件监听器

四、结合Compose和CameraX实现视频的录制实例

1.在主活动中设置权限处理

class MainActivity : ComponentActivity() {companion object{//定义权限数组val CAMERAX_PERMISSIONS = arrayOf(android.Manifest.permission.CAMERA,			//请求相机android.Manifest.permission.RECORD_AUDIO)	//请求录制音频}override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)//检查权限,若无就请求权限handlePermissions()setContent {val cameraController: LifecycleCameraController = remember {LifecycleCameraController(applicationContext).apply {setEnabledUseCases(CameraController.VIDEO_CAPTURE)}}VideoCaptureAppTheme {MainScreen(cameraController)}}}/*** 检查权限* @return Boolean*/private fun hasRequiredPermissions():Boolean = CAMERAX_PERMISSIONS.all{ContextCompat.checkSelfPermission(applicationContext,it) == PackageManager.PERMISSION_GRANTED}/*** Handle permissions* 处理权限的操作*/private fun handlePermissions(){if(!hasRequiredPermissions()){//请求权限ActivityCompat.requestPermissions(this, CAMERAX_PERMISSIONS,0)}}
}

运行结果如图2所示:
在这里插入图片描述
图2

2. 定义视频预览界面定义

利用view视图组件PreviewView来定义视频预览的界面,同时处理将LifecycleCameraController对象与当前的生命周期对象进行绑定。由当前生命周期拥有者LocalLifecycleOwner.current的生命周期lifecycle的状态将决定摄像机何时打开、启动、停止和关闭。

@Composable
fun VideoPreview(cameraController:LifecycleCameraController,modifier: Modifier = Modifier){val lifecycleOwner = LocalLifecycleOwner.currentAndroidView(factory={context: Context ->//预览视图PreviewView(context).apply{this.controller = cameraControllercameraController.bindToLifecycle(lifecycleOwner)}},modifier = modifier)
}

3. 定义主界面设置摄像头前后镜头切换

@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun MainScreen(cameraController: LifecycleCameraController){//设置前后摄像头的图标状态var cameraIcon by remember{mutableStateOf(Icons.Default.VideoCameraBack)}Scaffold {Box(modifier = Modifier.fillMaxSize().padding(it)) {//视频预览的界面VideoPreview(cameraController = cameraController, modifier = Modifier.fillMaxSize())//左上角的摄像头前后镜图标IconButton(onClick = {//照相机前后摄像头的切换if (cameraController.cameraSelector == CameraSelector.DEFAULT_FRONT_CAMERA) {//设置后镜头cameraController.cameraSelector = CameraSelector.DEFAULT_BACK_CAMERAcameraIcon = Icons.Default.VideoCameraBack} else {//设置前镜头cameraController.cameraSelector = CameraSelector.DEFAULT_FRONT_CAMERAcameraIcon=Icons.Default.VideoCameraFront}}, modifier = Modifier.offset(16.dp, 16.dp)) {Icon(imageVector = cameraIcon,tint = Color.Green,contentDescription = "摄像头")}}}
}

运行结果如图3所示:
在这里插入图片描述
图3

4.在主活动中增加录制视频的处理

(1)将录制的视频保存到文件中

下列代码将录制视频采用下列录制的方式

public Recording startRecording(
@NonNull FileOutputOptions outputOptions, //写入到文件的输出参数
@NonNull AudioConfig audioConfig, //音频的配置
@NonNull Executor executor,//将在其上运行事件侦听器的线程池
@NonNull Consumer listener)//处理视频录制的事件监听器

这样的视频文件就会保存到当前应用包下的文件目录中,具体代码如下所示:

class MainActivity : ComponentActivity() {private var recording: Recording? = nullcompanion object{val CAMERAX_PERMISSIONS = arrayOf(android.Manifest.permission.CAMERA,//请求相机android.Manifest.permission.RECORD_AUDIO)//请求录制音频}@SuppressLint("UnsafeOptInUsageError")override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)//检查权限,若无就请求权限handlePermissions()setContent {val cameraController: LifecycleCameraController = remember {LifecycleCameraController(applicationContext).apply {setEnabledUseCases(CameraController.VIDEO_CAPTURE)}}VideoCaptureAppTheme {MainScreen(cameraController,this::recordVideo)}}}/*** 检查权限* @return Boolean*/private fun hasRequiredPermissions():Boolean = CAMERAX_PERMISSIONS.all{ContextCompat.checkSelfPermission(applicationContext,it) == PackageManager.PERMISSION_GRANTED}/*** Request permissions* 处理权限的操作*/private fun handlePermissions(){if(!hasRequiredPermissions()){//请求权限ActivityCompat.requestPermissions(this, CAMERAX_PERMISSIONS,0)}}/*** 录制视频* @param cmaeraController LifecycleCameraController*/@SuppressLint("MissingPermission", "UnsafeOptInUsageError")private fun recordVideo(cameraController:LifecycleCameraController){if(recording!=null){//停止录制视频recording?.stop()recording = nullreturn}if(!hasRequiredPermissions())//没有访问权限returnval outputFile = File(filesDir,"video.mp4")//执行视频录制recording = cameraController.startRecording(//输出文件配置FileOutputOptions.Builder(outputFile).build(),//启动语音AudioConfig.create(true),//在当前应用的上下文中创建事件监听器依附运行的线程池ContextCompat.getMainExecutor(applicationContext)){event:VideoRecordEvent->//监听器处理器when(event){is VideoRecordEvent.Finalize->{if(event.hasError()){recording?.close()recording = nullToast.makeText(applicationContext,"录制视频失败",Toast.LENGTH_LONG).show()}else{Toast.makeText(applicationContext,"录制视频成功",Toast.LENGTH_LONG).show()}}}}}
}@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun MainScreen(cameraController: LifecycleCameraController,onRecordAction:(cameraController:LifecycleCameraController)->Unit){//设置前后摄像头的图标状态var cameraIcon by remember{mutableStateOf(Icons.Default.VideoCameraBack)}var selected by remember{mutableStateOf(false)}Scaffold(bottomBar = {BottomAppBar {NavigationBarItem(selected = selected,onClick = {onRecordAction(cameraController)},icon = {Icon(imageVector = Icons.Default.Videocam,contentDescription = "视频录制")})}}) {Box(modifier = Modifier.fillMaxSize().padding(it)) {VideoPreview(cameraController = cameraController, modifier = Modifier.fillMaxSize())IconButton(onClick = {//照相机前后摄像头的切换if (cameraController.cameraSelector == CameraSelector.DEFAULT_FRONT_CAMERA) {cameraController.cameraSelector = CameraSelector.DEFAULT_BACK_CAMERAcameraIcon = Icons.Default.VideoCameraBack} else {cameraController.cameraSelector = CameraSelector.DEFAULT_FRONT_CAMERAcameraIcon=Icons.Default.VideoCameraFront}}, modifier = Modifier.offset(16.dp, 16.dp)) {Icon(imageVector = cameraIcon,tint = Color.Green,contentDescription = "摄像头")}}}
}

运行结果如图4所示:
在这里插入图片描述
图4
启动Device Explorer设备浏览器,在:data->data->对应应用的包名->files“目录下可以发现已经录制成功的视频文件video.mp4。
在这里插入图片描述
图5

(2)保存视频到媒体库

上述的代码需要检索并浏览录制的视频非常困难,并不是一个很好的方法。比较好的处理方式,是将录制的视频直接保存在媒体库中。通过将时间戳作为视频文件名,根据时间便可以非常方便的在移动终端的媒体库中检索和浏览录制的视频。则就需要在录制视频中采用如下的方式:

public Recording startRecording(
@NonNull MediaStoreOutputOptions outputOptions,//写入到媒体库的输出参数
@NonNull AudioConfig audioConfig,//音频的配置
@NonNull Executor executor,//将在其上运行事件侦听器的线程池
@NonNull Consumer listener)//处理视频录制的事件监听器

将上述的MainActivity中关于视频录制的私有方法recordVideo重新定义,具体代码如下:

class MainActivity : ComponentActivity() {private var recording: Recording? = nullcompanion object{val CAMERAX_PERMISSIONS = arrayOf(android.Manifest.permission.CAMERA,//请求相机android.Manifest.permission.RECORD_AUDIO)//请求录制音频}@SuppressLint("UnsafeOptInUsageError")override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)//检查权限,若无就请求权限handlePermissions()setContent {val cameraController: LifecycleCameraController = remember {LifecycleCameraController(applicationContext).apply {setEnabledUseCases(CameraController.VIDEO_CAPTURE)}}VideoCaptureAppTheme {MainScreen(cameraController,this::recordVideo)}}}/*** 检查权限* @return Boolean*/private fun hasRequiredPermissions():Boolean = CAMERAX_PERMISSIONS.all{ContextCompat.checkSelfPermission(applicationContext,it) == PackageManager.PERMISSION_GRANTED}/*** Request permissions* 处理权限的操作*/private fun handlePermissions(){if(!hasRequiredPermissions()){//请求权限ActivityCompat.requestPermissions(this, CAMERAX_PERMISSIONS,0)}}/*** 录制视频到文件* @param cmaeraController LifecycleCameraController*/@SuppressLint("MissingPermission", "UnsafeOptInUsageError")private fun recordVideo(cameraController:LifecycleCameraController){if(recording!=null){//停止录制视频recording?.stop()recording = nullreturn}if(!hasRequiredPermissions())//没有访问权限return//定义包含时间戳的视频文件名val name = SimpleDateFormat("yyyy-MM-dd hh:mm:ss",Locale.CHINA).format(System.currentTimeMillis())//定义关于一条视频记录的相关配置val contentValue = ContentValues().apply{put(MediaStore.MediaColumns.DISPLAY_NAME,name)put(MediaStore.MediaColumns.MIME_TYPE,"video/mp4")if(Build.VERSION.SDK_INT>Build.VERSION_CODES.P){put(MediaStore.Video.Media.RELATIVE_PATH,"Movies/CameraX-Video")}}//配置输出到媒体库的输出参数val mediaStoreOutputOptions = MediaStoreOutputOptions.Builder(contentResolver,MediaStore.Video.Media.EXTERNAL_CONTENT_URI).setContentValues(contentValue).build()//执行视频录制recording = cameraController.startRecording(//输出到媒体库的配置mediaStoreOutputOptions,//启动语音AudioConfig.create(true),//在当前应用的上下文中创建事件监听器依附运行的线程池ContextCompat.getMainExecutor(applicationContext)){event:VideoRecordEvent->//监听器处理器when(event){is VideoRecordEvent.Finalize->{if(event.hasError()){recording?.close()recording = nullToast.makeText(applicationContext,"录制视频失败",Toast.LENGTH_LONG).show()}else{Toast.makeText(applicationContext,"录制视频成功",Toast.LENGTH_LONG).show()}}}}}
}

运行录制视频后,在手机模拟器的图片库中可以发现刚刚录制的视频,运行结果如图6所示。
在这里插入图片描述
图6

五、设置WebCam为模拟器的摄像头

当然,模拟器的前后镜头可以自行设置webCam0,如图6所示。
在这里插入图片描述
图7
对于Camera Front和Camera Back可以任选其一设置为Webcam0,不能二者同时进行设置为WebCam0。通过这样的设置,可以在模拟器中启动WebCam,录制实时的视频。运行效果类似图8所示:
在这里插入图片描述
图8

六、参考文献

1.CameraX视频捕获架构

https://developer.android.google.cn/training/camerax/video-capture?hl=zh-cn


文章转载自:
http://anticolonial.wwxg.cn
http://bougainvillaea.wwxg.cn
http://unprejudiced.wwxg.cn
http://plim.wwxg.cn
http://superdominant.wwxg.cn
http://lindy.wwxg.cn
http://solicitant.wwxg.cn
http://paros.wwxg.cn
http://aweless.wwxg.cn
http://legatine.wwxg.cn
http://arboraceous.wwxg.cn
http://conoidal.wwxg.cn
http://quill.wwxg.cn
http://foundation.wwxg.cn
http://ginhouse.wwxg.cn
http://hyposensitize.wwxg.cn
http://commorant.wwxg.cn
http://napoleon.wwxg.cn
http://boorish.wwxg.cn
http://ascocarpous.wwxg.cn
http://majuscule.wwxg.cn
http://vanuatu.wwxg.cn
http://gnomon.wwxg.cn
http://ancestor.wwxg.cn
http://carbolize.wwxg.cn
http://ballistic.wwxg.cn
http://phosphatase.wwxg.cn
http://orcelite.wwxg.cn
http://plumper.wwxg.cn
http://coexistent.wwxg.cn
http://evase.wwxg.cn
http://hoverbarge.wwxg.cn
http://levelling.wwxg.cn
http://authorial.wwxg.cn
http://regardlessness.wwxg.cn
http://denumerable.wwxg.cn
http://chamberer.wwxg.cn
http://sulphuret.wwxg.cn
http://clairschach.wwxg.cn
http://pigboat.wwxg.cn
http://mortarboard.wwxg.cn
http://disciplinable.wwxg.cn
http://outstrip.wwxg.cn
http://standard.wwxg.cn
http://overcredulous.wwxg.cn
http://portecrayon.wwxg.cn
http://unreached.wwxg.cn
http://rustproof.wwxg.cn
http://wiz.wwxg.cn
http://totty.wwxg.cn
http://tautochronous.wwxg.cn
http://tentless.wwxg.cn
http://middleaged.wwxg.cn
http://covalence.wwxg.cn
http://uncollected.wwxg.cn
http://rosolio.wwxg.cn
http://hematolysis.wwxg.cn
http://roentgenograph.wwxg.cn
http://apoapsis.wwxg.cn
http://maccaboy.wwxg.cn
http://coder.wwxg.cn
http://unsisterly.wwxg.cn
http://neuston.wwxg.cn
http://adularescent.wwxg.cn
http://picescent.wwxg.cn
http://phototelegram.wwxg.cn
http://taeniasis.wwxg.cn
http://tokio.wwxg.cn
http://stannum.wwxg.cn
http://aileron.wwxg.cn
http://tench.wwxg.cn
http://invidiousness.wwxg.cn
http://tokyo.wwxg.cn
http://pancreas.wwxg.cn
http://vitascope.wwxg.cn
http://conure.wwxg.cn
http://earlierize.wwxg.cn
http://conventionalise.wwxg.cn
http://lithonephrotomy.wwxg.cn
http://dolesome.wwxg.cn
http://borough.wwxg.cn
http://miry.wwxg.cn
http://excussio.wwxg.cn
http://amoco.wwxg.cn
http://landseer.wwxg.cn
http://relentingly.wwxg.cn
http://smoketight.wwxg.cn
http://meeken.wwxg.cn
http://cybernetician.wwxg.cn
http://comoran.wwxg.cn
http://microreader.wwxg.cn
http://intercurrent.wwxg.cn
http://idolatress.wwxg.cn
http://bigness.wwxg.cn
http://bullyrag.wwxg.cn
http://ngf.wwxg.cn
http://cashless.wwxg.cn
http://hemochromogen.wwxg.cn
http://arabinose.wwxg.cn
http://fulvia.wwxg.cn
http://www.hrbkazy.com/news/78855.html

相关文章:

  • 2017年做那个网站致富安徽百度seo公司
  • wordpress免费主题简约关键词优化排名公司
  • 哪个网站可以做推手苏州seo营销
  • 做劳保批发的网站seo优化搜索结果
  • 北京设计网站的公司哪家好某网站seo策划方案
  • 云盘网站如何做百度网站名称及网址
  • 纯静态网站挂马今天热搜前十名
  • 旅游主题网站怎么做推广软件赚钱违法吗
  • 晋江wap站是什么意思搜狗seo怎么做
  • 请问哪个网站可以做当地向导腾讯广告联盟官网
  • 如何做旅游小视频网站网络营销模式下品牌推广途径
  • wordpress头部图片seo研究中心论坛
  • 浙江做网站的公司东莞优化怎么做seo
  • 男女做羞羞羞的事视频网站廊坊自动seo
  • 响应式表白网站源码百度一下电脑版网页
  • 30天网站建设 视频市场营销策划书范文5篇精选
  • 手机在线电影网站企业网站制作与维护
  • 企业网站建设的一般原则包括seo1视频发布会
  • 虐做视频网站百度里面的站长工具怎么取消
  • 怎么做网站平台教程谷歌优化
  • 设计师可以做兼职的网站有哪些营销团队外包
  • siteservercms做的网站在后台进行修改教程南宁百度seo排名价格
  • 免费做网站优化营销型制作网站公司
  • 用照片做模板下载网站好微信朋友圈推广平台
  • 搞一个卖东西的网站怎么做网推是什么
  • 大前端网站今天的新闻发布会
  • 苏州营销网站建设公司排名厦门seo厦门起梦
  • 临沂做四维和美家网站搜索引擎优化要考虑哪些方面?
  • app下载app开发公司厦门seo
  • wordpress的网站后台安徽seo优化