一、需求
1.使用kotlin进行PermissionsDispatcher动态权限管理
2.卡片式RecycleView的实现
二、实现步骤
实现第一个需求
1.添加依赖
app模块的build.gradle
dependencies{
...
implementation "org.permissionsdispatcher:permissionsdispatcher:4.7.0"
kapt "org.permissionsdispatcher:permissionsdispatcher-processor:4.7.0"
}
2.使用注解
@RuntimePermissions | 注解在需要用到权限的Activity或者Fragment上 |
---|---|
@NeedPermission | 注解在用户同意权限后需要执行的方法上 |
– | – |
@OnShowRationale | 注解在用户第一次拒绝权限,下次请求权限前会调用的方法上 |
– | – |
@OnPermissionDenied | 注解在用户拒绝了权限请求时需要调用的方法上 |
– | – |
@OnShowRationale | 注解在当用户选中了授权窗口中的不再询问复选框后并拒绝了权限请求时需要调用的方法,一般可以向用户解释为何申请此权限,并根据实际需求决定是否再次弹出权限请求对话框 |
**
特别注意:被注解的方法不能是私有方法(private)
原因:因为用了注解的方法或者类,是根据你的代码使用第三方库生成代码,如果你的代码是私有的,那么生成的代码也会是私有的,这时候你的类就无法拿到生成的私有代码,所有出错
不然会报错 我是出现这样的错误,当时解决了好久 Failed to notify project evaluation listener.
java.lang.reflect.InvocationTargetException (no error message)
**
这些注解中@RuntimePermissions和@NeedsPermission是必须有的,其他注解根据自己的情况选择
我们在需要的类或者方法上写好注解后,需要点击菜单栏中 Build 菜单下的 Make Project ,或者按快捷键 Ctrl + F9 编译整个项目
因为我用的是kotlin语言,所以我是在java(generated)文件中生成MineFragmentPermissionDispatcher.kt文件,用Java语言文件生成的位置好像不一样。
这个名字的组成是由我的类名加PermissionDispatcher生成的
如果要使用被@NeedPermission注解的方法,这时候就要这样写
openAlumWithPermissionCheck()
这是被注解生成的方法,被注解的方法名是
@NeedsPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE)
fun openAlum(){
val intent = Intent(Intent.ACTION_PICK, null)
intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*")
startActivityForResult(intent, CHOOSE_PHOTO)
popupWindow?.dismiss()
}
然后就是重写Activity或者Fragment的onRequestPermissionResult()方法
override fun onRequestPermissionsResult(
requestCode: Int,
permissions: Array<out String>,
grantResults: IntArray
) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
onRequestPermissionsResult(requestCode, grantResults)
}
卡片式RecyclerView的实现
1.添加依赖
implementation 'com.github.DingMouRen:LayoutManagerGroup:1e6f4f96eb'
2.写recycleview中item的布局
<?xml version="1.0" encoding="utf-8"?>
<androidx.cardview.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/card_view_sport"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="10dp"
app:cardCornerRadius="10dp"
app:elevation="10dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:background="@drawable/sport_item_bg">
<ImageView
android:id="@+id/img_sport_item_tu"
android:layout_width="match_parent"
android:layout_height="400dp"
android:src="@drawable/majiaxian"
android:scaleType="fitXY"/>
<Button
android:id="@+id/btn_sport_item_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="马甲线"
android:background="@drawable/sport_item_btn_bg"
android:layout_margin="20dp"
android:layout_gravity="center"
android:textColor="@color/white"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
3.用到recyclerview的Activity布局
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/color_view_bg"
tools:context=".ui.activity.SportActivity"
android:padding="20dp">
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/recycle_sport_card"
android:layout_width="match_parent"
android:layout_height="500dp"
android:layout_centerInParent="true"/>
</RelativeLayout>
4.写recyclerview的适配器
5.在用到recycleview的活动中使用适配器,是数据和view结合起来
这里就用到了我们依赖的东西了
val recycler = findViewById<RecyclerView>(R.id.recycle_sport_card)
mSkidRightLayoutManager = SkidRightLayoutManager(1.7f,0.85f)
recycler.layoutManager = mSkidRightLayoutManager
recycler.adapter = mAdapter
就是设置recycleview.layoutmanager用到的是SkidRightLayoutManager(1.7f,0.85f)
导入这样写
import com.dingmouren.layoutmanagergroup.skidright.SkidRightLayoutManager