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

Compose笔记(三十)--图片选择器

        这一节主要了解一下Compose中的图片选择器,图片选择器是一种允许用户从设备相册或文件系统中选择图片的组件,它简化了媒体文件访问流程,简单总结如下:

API:
ActivityResultContracts.GetMultipleContents()
用于从系统选择多个媒体文件的契约(Contract)。它允许用户通过系统文件选择器同时选择多张图片、视频或其他类型的文件,并返回这些文件的Uri列表。
ActivityResultContracts.PickMultipleVisualMedia()
Android13(API 33)引入的专用契约,用于从系统媒体库中多选图片或视频。它提供了更优化的媒体选择体验,自动处理权限请求,并支持按类型过滤。

 

栗子:

 

package com.example.test0121.imagepickerimport android.net.Uri
import android.os.Build
import androidx.activity.compose.rememberLauncherForActivityResult
import androidx.activity.result.PickVisualMediaRequest
import androidx.activity.result.contract.ActivityResultContracts
import androidx.compose.foundation.background
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.*
import androidx.compose.foundation.shape.CircleShape
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.material3.*
import androidx.compose.runtime.*
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.draw.clip
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.layout.ContentScale
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.painterResource
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import coil.compose.AsyncImage
import coil.request.ImageRequest
import com.google.accompanist.permissions.ExperimentalPermissionsApi@OptIn(ExperimentalPermissionsApi::class, ExperimentalLayoutApi::class)
@Composable
fun ImagePickerDemo() {val context = LocalContext.currentval maxImages = 6var selectedImages by remember { mutableStateOf<List<Uri>>(emptyList()) }val isAndroid15OrHigher = Build.VERSION.SDK_INT >= Build.VERSION_CODES.UPSIDE_DOWN_CAKEval mediaPickerLauncher = rememberLauncherForActivityResult(contract = ActivityResultContracts.PickMultipleVisualMedia()) { uris ->selectedImages = uris}val legacyImagePickerLauncher = rememberLauncherForActivityResult(contract = ActivityResultContracts.GetMultipleContents()) { uris ->selectedImages = uris.take(maxImages)}Column(modifier = Modifier.fillMaxSize().padding(16.dp),horizontalAlignment = Alignment.CenterHorizontally) {Text(text = "多图选择器测试",fontSize = 20.sp,fontWeight = FontWeight.Bold,modifier = Modifier.padding(bottom = 16.dp))FlowRow(modifier = Modifier.fillMaxWidth(),horizontalArrangement = Arrangement.spacedBy(8.dp),verticalArrangement = Arrangement.spacedBy(8.dp)) {if (selectedImages.size < maxImages) {Box(modifier = Modifier.size(100.dp).clip(RoundedCornerShape(8.dp)).background(MaterialTheme.colorScheme.surfaceVariant).clickable {if (isAndroid15OrHigher) {mediaPickerLauncher.launch(PickVisualMediaRequest(mediaType = ActivityResultContracts.PickVisualMedia.ImageOnly,))} else {                             legacyImagePickerLauncher.launch("image/*")}},contentAlignment = Alignment.Center) {Column(horizontalAlignment = Alignment.CenterHorizontally) {Icon(painter = painterResource(android.R.drawable.ic_menu_camera),contentDescription = "添加图片",tint = MaterialTheme.colorScheme.onSurfaceVariant,modifier = Modifier.size(30.dp))Text(text = "添加图片",fontSize = 12.sp,color = MaterialTheme.colorScheme.onSurfaceVariant,modifier = Modifier.padding(top = 4.dp))}}}selectedImages.forEachIndexed { index, uri ->Box(modifier = Modifier.size(100.dp)) {AsyncImage(model = ImageRequest.Builder(context).data(uri).crossfade(true).build(),contentDescription = "选择的图片 $index",contentScale = ContentScale.Crop,modifier = Modifier.fillMaxSize().clip(RoundedCornerShape(8.dp)))IconButton(onClick = { selectedImages = selectedImages.filterIndexed { i, _ -> i != index } },modifier = Modifier.size(28.dp).align(Alignment.TopEnd).background(Color.Black.copy(alpha = 0.5F), CircleShape).padding(2.dp)) {Icon(painter = painterResource(android.R.drawable.ic_delete),contentDescription = "删除",tint = Color.White,modifier = Modifier.size(16.dp))}}}}Text(text = "已选择 ${selectedImages.size}/$maxImages 张图片",fontSize = 14.sp,color = MaterialTheme.colorScheme.secondary,modifier = Modifier.padding(top = 12.dp))}
}
清单文件:<uses-permissionandroid:name="android.permission.WRITE_EXTERNAL_STORAGE"tools:ignore="QueryAllPackagesPermission" /> <!-- <uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" --><!-- tools:ignore="ScopedStorage" /> --><uses-permission android:name="android.permission.READ_MEDIA_IMAGES" /> <uses-permission android:name="android.permission.READ_MEDIA_VISUAL_USER_SELECTED" /> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /><uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

http://www.dtcms.com/a/266788.html

相关文章:

  • 【Spring Boot】HikariCP 连接池 YAML 配置详解
  • 洛谷P1941 [NOIP 2014 提高组] 飞扬的小鸟
  • vue3 获取选中的el-table行数据
  • MySQL 查询进阶指南:子查询、多表连接与 UNION 操作全解析
  • SQL 快速参考手册-SQL001
  • Swagger 安装使用教程
  • 高效的在Vue3中使用Vuex
  • Android-自定义View的实战学习总结
  • python训练day49 CBAM
  • 流程分类框架体系设计应该梳理到L5还是L6?
  • DePIN 普惠结构的缺失拼图,为什么是 UBI Network?
  • js中的捕获阶段和冒泡阶段
  • vue2/3安装依赖报错,终极解决方案
  • Kuberrnetes 服务发布
  • 【MySQL】十六,MySQL窗口函数
  • Mint密室 · 猫猫狐狐的“特征选择”囚室逃脱
  • Ubuntu下的Tomcat服务器部署
  • Linux基础 -- NAND Flash UBIFS基础特性及注意点
  • 【沉浸式解决问题】idea开发中mapper类中突然找不到对应实体类
  • 【Agent】构建专家级SQL Agent交互
  • Qt控件核心属性全解析
  • 【Bluedroid】 BLE 隐私保护机制深度剖析(btm_ble_reset_id)
  • [学习记录]Unity-Shader-曲面细分着色器
  • IDEA-常用的开发组件
  • 用户进程的借壳挂靠之术
  • JAVA-springboot 整合Redis
  • 大数据在UI前端的应用创新研究:基于图神经网络的用户关系网络分析
  • [C++] C++多重继承:深入解析复杂继承关系
  • Blob分析及形态学分析
  • AWS 中如何添加一个内部域名