1、定义recyclerview的布局dialog_custom_list.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:padding="@dimen/_10sdp">
<TextView
android:id="@+id/tv_title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:textColor="@color/grey_900"
android:textSize="@dimen/_15sdp"
android:textStyle="bold"
android:gravity="center"
android:layout_gravity="center"
tools:text="Dialog Title"/>
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/rv_list"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="@dimen/_10sdp"
android:visibility="visible"
tools:visibility="visible">
</androidx.recyclerview.widget.RecyclerView>
</LinearLayout>
2、定义每个条目的布局 item_custom_list.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<TextView
android:id="@+id/tv_text"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="?attr/selectableItemBackground"
android:gravity="center"
android:padding="@dimen/_10sdp"
android:textAllCaps="true"
android:textColor="@color/grey_900"
android:textSize="@dimen/_16sdp"
android:text="Item Value"/>
<View
android:layout_width="match_parent"
android:layout_height="@dimen/_1sdp"
android:background="@color/divider_line_color"/>
</LinearLayout>
3、准备recycle人view中所需要的数据
package com.example.mykotlin1
object Constants {
const val DISH_TYPE : String = "DishType"
const val DISH_CATEGORY : String = "DishCategory"
const val DISH_COOKING_TIME : String = "DishCookingTime"
fun dishTypes():ArrayList<String>{
val list = ArrayList<String>()
list.add("breakfast")
list.add("launch")
list.add("snacks")
list.add("dinner")
list.add("salad")
list.add("side dish")
list.add("dessert")
list.add("other")
return list
}
fun dishCategories():ArrayList<String>{
val list = ArrayList<String>()
list.add("Pizza")
list.add("BBQ")
list.add("Bakery")
list.add("Burger")
list.add("Cafe")
list.add("Chicker")
list.add("Dessert")
list.add("Drinks")
list.add("Hot Dogs")
list.add("Juices")
list.add("Sandwich")
list.add("Tea & Coffee")
list.add("Wraps")
list.add("Others")
return list
}
fun dishCookTime():ArrayList<String>{
val list = ArrayList<String>()
list.add("10")
list.add("15")
list.add("20")
list.add("30")
list.add("45")
list.add("50")
list.add("60")
list.add("90")
list.add("120")
list.add("150")
list.add("180")
list.add("Wraps")
list.add("Others")
return list
}
}
4、加载包含recyclerview的布局
val binding : DialogCustomListBinding = DialogCustomListBinding.inflate(layoutInflater)
customListDialog.setContentView(binding.root)
5、设置recyclerview的layoutmanager
binding.rvList.layoutManager = LinearLayoutManager(this)
6、设置recyclerview的adapter
val adapter = CustomListItemAdapter(this,itemsList,selection)
binding.rvList.adapter = adapter
界面显示:
注:
activity的代码:
package com.example.mykotlin1
import android.Manifest
import android.app.Activity
import android.app.AlertDialog
import android.app.Dialog
import android.content.ActivityNotFoundException
import android.content.Context
import android.content.ContextWrapper
import android.content.Intent
import android.graphics.Bitmap
import android.graphics.drawable.Drawable
import android.net.Uri
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.provider.MediaStore
import android.provider.Settings
import android.util.Log
import android.view.View
import android.widget.Toast
import androidx.core.content.ContextCompat
import androidx.core.graphics.drawable.toBitmap
import androidx.recyclerview.widget.LinearLayoutManager
import com.bumptech.glide.Glide
import com.bumptech.glide.load.DataSource
import com.bumptech.glide.load.engine.DiskCacheStrategy
import com.bumptech.glide.load.engine.GlideException
import com.bumptech.glide.request.RequestListener
import com.bumptech.glide.request.target.Target
import com.example.mykotlin1.adapters.CustomListItemAdapter
import com.example.mykotlin1.databinding.ActivityAddUpdateDishBinding
import com.example.mykotlin1.databinding.DialogCustomImageSelectionBinding
import com.example.mykotlin1.databinding.DialogCustomListBinding
import com.karumi.dexter.Dexter
import com.karumi.dexter.MultiplePermissionsReport
import com.karumi.dexter.PermissionToken
import com.karumi.dexter.listener.PermissionDeniedResponse
import com.karumi.dexter.listener.PermissionGrantedResponse
import com.karumi.dexter.listener.PermissionRequest
import com.karumi.dexter.listener.multi.MultiplePermissionsListener
import com.karumi.dexter.listener.single.PermissionListener
import java.io.File
import java.io.FileOutputStream
import java.io.IOException
import java.io.OutputStream
import java.util.*
class AddUpdateDishActivity : AppCompatActivity(), View.OnClickListener {
private lateinit var mBinding : ActivityAddUpdateDishBinding
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
mBinding = ActivityAddUpdateDishBinding.inflate(layoutInflater)
setContentView(mBinding.root)
mBinding.ivAddDishImage.setOnClickListener(this)
mBinding.etType.setOnClickListener(this)
mBinding.etCategory.setOnClickListener(this)
mBinding.etCookingTime.setOnClickListener(this)
// mBinding.svContainer.setHorizontalFadingEdgeEnabled(false)
//setupActionBar()
}
private fun setupActionBar(){
val toobar = mBinding.toolbarAddDishActivity
// Log.i("AddUpdateDishActivity",toobar.isEmpty()+"")
setSupportActionBar(mBinding.toolbarAddDishActivity)
supportActionBar?.setDisplayHomeAsUpEnabled(true)
mBinding.toolbarAddDishActivity.setNavigationOnClickListener {
onBackPressed()
}
}
override fun onClick(v: View?) {
when(v!!.id){
R.id.iv_add_dish_image ->{
customImageSelectionDialog()
return;
}
R.id.et_type ->{
customItemsListDialog(resources.getString(R.string.title_select_dish_type),
Constants.dishTypes(),
Constants.DISH_TYPE)
return
}
R.id.et_category ->{
customItemsListDialog(resources.getString(R.string.title_select_dish_category),
Constants.dishCategories(),
Constants.DISH_CATEGORY)
return
}
R.id.et_cooking_time ->{
customItemsListDialog(resources.getString(R.string.title_select_dish_cooking_time),
Constants.dishCookTime(),
Constants.DISH_COOKING_TIME)
return
}
}
}
private fun customImageSelectionDialog(){
val dialog = Dialog(this);
val binding : DialogCustomImageSelectionBinding = DialogCustomImageSelectionBinding.inflate(layoutInflater)
dialog.setContentView(binding.root)
binding.tvCamera.setOnClickListener {
Dexter.withContext(this).withPermissions(
Manifest.permission.READ_EXTERNAL_STORAGE,
//Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.CAMERA
).withListener(object : MultiplePermissionsListener {
override fun onPermissionsChecked(report : MultiplePermissionsReport?) {
report?.let {
if(report!!.areAllPermissionsGranted()){
val intent = Intent(MediaStore.ACTION_IMAGE_CAPTURE)
startActivityForResult(intent,CAMERA)
}
}
}
override fun onPermissionRationaleShouldBeShown(
permissions: MutableList<PermissionRequest>?,
token: PermissionToken?
) {
showRationalDialogForPermissions()
}
}).onSameThread().check()
dialog.dismiss()
}
binding.tvGallery.setOnClickListener {
Dexter.withContext(this@AddUpdateDishActivity)
.withPermission(
Manifest.permission.READ_EXTERNAL_STORAGE,
).withListener(object : PermissionListener {
override fun onPermissionGranted(p0: PermissionGrantedResponse?) {
val galleryIntent = Intent(Intent.ACTION_PICK,MediaStore.Images.Media.EXTERNAL_CONTENT_URI)
startActivityForResult(galleryIntent, GALLERY)
}
override fun onPermissionDenied(p0: PermissionDeniedResponse?) {
Toast.makeText(this@AddUpdateDishActivity,"You have denied to storage permission to select image",Toast.LENGTH_SHORT).show()
}
override fun onPermissionRationaleShouldBeShown(
p0: PermissionRequest?,
p1: PermissionToken?
) {
showRationalDialogForPermissions()
}
}).onSameThread().check()
dialog.dismiss()
}
dialog.show()
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
if(resultCode == Activity.RESULT_OK){
if(requestCode == CAMERA){
data?.extras?.let {
val thumbnail : Bitmap = data.extras!!.get("data") as Bitmap
// mBinding.ivDishImage.setImageBitmap(thumbnail)
Glide.with(this)
.load(thumbnail)
.centerCrop()
.into(mBinding.ivDishImage)
mImagePath = saveImageToInternalStorage(thumbnail )
Log.e("cancelled","User mImagePath ------>"+mImagePath)
mBinding.ivAddDishImage.setImageDrawable(ContextCompat.getDrawable(this,R.drawable.ic_vector_edit))
}
}
if(requestCode == GALLERY){
data?.data?.let {
val selectedPhotoUri = data.data
// mBinding.ivDishImage.setImageURI(selectedPhotoUri)
Glide.with(this)
.load(selectedPhotoUri)
.centerCrop()
.diskCacheStrategy(DiskCacheStrategy.ALL)
.listener(object : RequestListener<Drawable>{
override fun onLoadFailed(
e: GlideException?,
model: Any?,
target: Target<Drawable>?,
isFirstResource: Boolean
): Boolean {
Log.e("cancelled","Error loading image",e)
return false;
}
override fun onResourceReady(
resource: Drawable?,
model: Any?,
target: Target<Drawable>?,
dataSource: DataSource?,
isFirstResource: Boolean
): Boolean {
resource.let {
val bitmap : Bitmap = resource!!.toBitmap()
mImagePath = saveImageToInternalStorage(bitmap)
Log.e("cancelled","User mImagePath ------>"+mImagePath)
}
return false
}
})
.into(mBinding.ivDishImage)
mBinding.ivAddDishImage.setImageDrawable(ContextCompat.getDrawable(this,R.drawable.ic_vector_edit))
}
}
}else if( resultCode == Activity.RESULT_CANCELED){
Log.e("cancelled","User canceled selection")
}
}
private fun showRationalDialogForPermissions(){
AlertDialog.Builder(this).setMessage("It looks like you have turned off permissions " +
"required for this feature.It can be enabled under Application Settings")
.setPositiveButton("GO TO SETTINGS")
{_,_ ->
try{
val intent = Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS)
val uri = Uri.fromParts("package",packageName,null)
intent.data = uri
startActivity(intent)
}catch (e : ActivityNotFoundException){
e.printStackTrace()
}
}.setNegativeButton("Cancel"){dialog,_->
dialog.dismiss()
}.show()
}
private var mImagePath : String = ""
private fun saveImageToInternalStorage(bitmap: Bitmap):String{
val wrapper = ContextWrapper(applicationContext)
var file = wrapper.getDir(IMAGE_DIRECTORY, Context.MODE_PRIVATE)
file = File(file,"${UUID.randomUUID()}.jpg")
try {
val stream : OutputStream = FileOutputStream(file)
bitmap.compress(Bitmap.CompressFormat.JPEG,100,stream)
stream.flush()
stream.close()
}catch (e:IOException){
e.printStackTrace()
}
return file.absolutePath
}
private fun customItemsListDialog(title : String, itemsList : List<String>, selection : String){
val customListDialog = Dialog(this)
val binding : DialogCustomListBinding = DialogCustomListBinding.inflate(layoutInflater)
customListDialog.setContentView(binding.root)
binding.tvTitle.text = title
binding.rvList.layoutManager = LinearLayoutManager(this)
val adapter = CustomListItemAdapter(this,itemsList,selection)
binding.rvList.adapter = adapter
customListDialog.show()
}
companion object{
private const val CAMERA = 1
private const val GALLERY = 2
private const val IMAGE_DIRECTORY = "FavDishImages"
}
}