├── .gitattributes ├── .gitignore ├── README.md ├── Screenshot_20191208-160231_PolicyLibDemo.jpg ├── Screenshot_20191208-160310_Package installer.jpg ├── Screenshot_20191221-185304_PolicyLibDemo.jpg ├── Screenshot_20191221-185325_PolicyLibDemo.jpg ├── Screenshot_20191221-185419_PolicyLibDemo.jpg ├── Screenshot_20191230-201024_PolicyLibDemo.jpg ├── app-debug.apk ├── app ├── .gitignore ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── db │ │ └── policylibdemo │ │ └── ExampleInstrumentedTest.java │ ├── main │ ├── AndroidManifest.xml │ ├── assets │ │ ├── privateRule.html │ │ └── userRule.html │ ├── java │ │ └── com │ │ │ └── db │ │ │ └── policylibdemo │ │ │ ├── BaseApplication.java │ │ │ ├── MainActivity.java │ │ │ ├── MainFragment.java │ │ │ └── RuleActivity.java │ └── res │ │ ├── drawable-v24 │ │ └── ic_launcher_foreground.xml │ │ ├── drawable │ │ └── ic_launcher_background.xml │ │ ├── layout │ │ ├── activity_main.xml │ │ ├── activity_rule.xml │ │ └── fragment_main.xml │ │ ├── mipmap-anydpi-v26 │ │ ├── ic_launcher.xml │ │ └── ic_launcher_round.xml │ │ ├── mipmap-hdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-mdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xhdpi │ │ ├── ic_launcher.png │ │ ├── ic_launcher_round.png │ │ └── icon_back_arrow.png │ │ ├── mipmap-xxhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xxxhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── values │ │ ├── colors.xml │ │ ├── strings.xml │ │ └── styles.xml │ │ └── xml │ │ ├── file_paths.xml │ │ └── net_config.xml │ └── test │ └── java │ └── com │ └── db │ └── policylibdemo │ └── ExampleUnitTest.java ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── local.properties ├── policylib-X-release.aar ├── policylib-release.aar ├── policylib ├── .gitignore ├── build.gradle ├── consumer-rules.pro ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── db │ │ └── policylib │ │ └── ExampleInstrumentedTest.java │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ ├── com │ │ │ └── db │ │ │ │ └── policylib │ │ │ │ ├── PermissionPolicy.java │ │ │ │ ├── Policy.java │ │ │ │ ├── PolicyAdapter.java │ │ │ │ ├── UpdateUtils.java │ │ │ │ ├── permission │ │ │ │ ├── PermissionFragment.java │ │ │ │ └── PermissionSuit.java │ │ │ │ └── receiver │ │ │ │ └── DownLoadReceiver.java │ │ └── pub │ │ │ └── devrel │ │ │ └── easypermissions │ │ │ ├── AfterPermissionGranted.java │ │ │ ├── AppSettingsDialog.java │ │ │ ├── AppSettingsDialogHolderActivity.java │ │ │ ├── EasyPermissions.java │ │ │ ├── PermissionRequest.java │ │ │ ├── RationaleClickListener.java │ │ │ ├── RationaleDialogClickListener.java │ │ │ ├── RationaleDialogConfig.java │ │ │ ├── RationaleDialogFragment.java │ │ │ ├── RationaleDialogFragmentCompat.java │ │ │ └── helper │ │ │ ├── ActivityPermissionHelper.java │ │ │ ├── AppCompatActivityPermissionsHelper.java │ │ │ ├── BaseSupportPermissionsHelper.java │ │ │ ├── LowApiPermissionsHelper.java │ │ │ ├── PermissionHelper.java │ │ │ ├── SupportFragmentPermissionHelper.java │ │ │ └── package-info.java │ └── res │ │ ├── drawable │ │ ├── btn_normal.xml │ │ ├── btn_pressed.xml │ │ ├── btn_rule.xml │ │ ├── btn_rule_normal.xml │ │ ├── btn_rule_pressed.xml │ │ ├── btn_rule_right_normal.xml │ │ ├── btn_rule_right_pressed.xml │ │ ├── btn_rule_right_selector.xml │ │ ├── btn_rule_selector.xml │ │ └── btn_selector.xml │ │ ├── layout │ │ ├── item_policy.xml │ │ ├── layout_policy.xml │ │ ├── layout_rule.xml │ │ └── layout_update.xml │ │ ├── mipmap-xhdpi │ │ ├── icon_camera_permission.png │ │ ├── icon_record_audio.png │ │ ├── icon_storage.png │ │ └── icon_tel.png │ │ ├── values │ │ ├── colors.xml │ │ ├── strings.xml │ │ └── styles.xml │ │ └── xml │ │ ├── file_paths.xml │ │ └── net_config.xml │ └── test │ └── java │ └── com │ └── db │ └── policylib │ └── ExampleUnitTest.java └── settings.gradle /.gitattributes: -------------------------------------------------------------------------------- 1 | # Auto detect text files and perform LF normalization 2 | * text=auto 3 | *.js linguist-language=java 4 | *.css linguist-language=java 5 | *.html linguist-language=java 6 | *.java linguist-language=java 7 | # Custom for Visual Studio 8 | *.cs diff=csharp 9 | 10 | # Standard to msysgit 11 | *.doc diff=astextplain 12 | *.DOC diff=astextplain 13 | *.docx diff=astextplain 14 | *.DOCX diff=astextplain 15 | *.dot diff=astextplain 16 | *.DOT diff=astextplain 17 | *.pdf diff=astextplain 18 | *.PDF diff=astextplain 19 | *.rtf diff=astextplain 20 | *.RTF diff=astextplain 21 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.class 2 | 3 | # BlueJ files 4 | *.ctxt 5 | 6 | # Mobile Tools for Java (J2ME) 7 | .mtj.tmp/ 8 | 9 | # Package Files # 10 | *.jar 11 | *.war 12 | *.ear 13 | 14 | # virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml 15 | hs_err_pid* 16 | 17 | # ========================= 18 | # Operating System Files 19 | # ========================= 20 | 21 | # Windows 22 | # ========================= 23 | 24 | # Windows thumbnail cache files 25 | Thumbs.db 26 | ehthumbs.db 27 | ehthumbs_vista.db 28 | 29 | # Folder config file 30 | Desktop.ini 31 | 32 | # Recycle Bin used on file shares 33 | $RECYCLE.BIN/ 34 | 35 | # Windows Installer files 36 | *.cab 37 | *.msi 38 | *.msm 39 | *.msp 40 | 41 | # Windows shortcuts 42 | *.lnk 43 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # PolicyLib 2 | PolicyLib 3 | 4 | 权限申请和隐私政策弹窗提示库: 5 | 6 | 支持工信部规定的首次弹窗隐私政策和用户协议; 7 | 支持申请权限前弹窗提示说明权限用途; 8 | 支持申请权限时,用户点击拒绝并不再提示后再弹窗提示去系统手动授权; 9 | 内置权限申请库:EasyPermission; 10 | 支持升级更新功能:UpdateUtils。 11 | 12 | 具体用例可以在MainActivity里查看用法。支持Activity和Fragment。 13 | 14 | 可以直接使用提供的policylib-release.aar,AndroidX版本请使用:policylib-X-release.aar。 15 | 16 | 体验APK: 17 | [点击下载](https://github.com/jaychou2012/PolicyLibDemo/blob/master/app-debug.apk?raw=true) 18 | 19 | ![隐私政策和用户协议首次弹窗](https://github.com/jaychou2012/PolicyLibDemo/blob/master/Screenshot_20191208-160231_PolicyLibDemo.jpg?raw=true) 20 | 21 | 22 | ![申请权限前弹窗提示说明权限用途](https://github.com/jaychou2012/PolicyLibDemo/blob/master/Screenshot_20191221-185304_PolicyLibDemo.jpg?raw=true) 23 | 24 | 25 | ![权限申请](https://github.com/jaychou2012/PolicyLibDemo/blob/master/Screenshot_20191208-160310_Package%20installer.jpg?raw=true) 26 | 27 | ![权限不再提示](https://github.com/jaychou2012/PolicyLibDemo/blob/master/Screenshot_20191221-185419_PolicyLibDemo.jpg?raw=true) 28 | 29 | ![升级功能](https://github.com/jaychou2012/PolicyLibDemo/blob/master/Screenshot_20191230-201024_PolicyLibDemo.jpg?raw=true) 30 | 31 | 32 | 33 | ## 《Android开发进阶实战:拓展与提升》已出版 34 | 35 | 36 | ### 新书涵盖Android最新的技术和内容,包括:新布局方式ConstraintLayout 、AndroidX、Jetpack、TV开发等,值得购买阅读。 37 | 38 | 39 | ![Android开发进阶实战:拓展与提升](http://img13.360buyimg.com/n1/jfs/t1/113550/10/7905/112523/5ec79791E6bf5d507/7169944c4d0d6669.jpg "Android开发进阶实战:拓展与提升") 40 | 41 | 42 | ### 纸质书购买: 43 | 44 | [京东](https://item.jd.com/69496918930.html "京东") [天猫](https://detail.tmall.com/item.htm?spm=a220m.1000858.1000725.6.7103434dRkHC8S&id=618745314823&user_id=3446196188&cat_id=2&is_b=1&rn=45bd1618b102199a8f9794a7b8431df4 "天猫") [当当](http://product.dangdang.com/28552590.html "当当") 45 | 46 | 47 | 48 | -------------------------------------------------------------------------------- /Screenshot_20191208-160231_PolicyLibDemo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jaychou2012/PolicyLib/d894e4ce3ffbe82b06624c8d731df0833ef51ca9/Screenshot_20191208-160231_PolicyLibDemo.jpg -------------------------------------------------------------------------------- /Screenshot_20191208-160310_Package installer.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jaychou2012/PolicyLib/d894e4ce3ffbe82b06624c8d731df0833ef51ca9/Screenshot_20191208-160310_Package installer.jpg -------------------------------------------------------------------------------- /Screenshot_20191221-185304_PolicyLibDemo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jaychou2012/PolicyLib/d894e4ce3ffbe82b06624c8d731df0833ef51ca9/Screenshot_20191221-185304_PolicyLibDemo.jpg -------------------------------------------------------------------------------- /Screenshot_20191221-185325_PolicyLibDemo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jaychou2012/PolicyLib/d894e4ce3ffbe82b06624c8d731df0833ef51ca9/Screenshot_20191221-185325_PolicyLibDemo.jpg -------------------------------------------------------------------------------- /Screenshot_20191221-185419_PolicyLibDemo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jaychou2012/PolicyLib/d894e4ce3ffbe82b06624c8d731df0833ef51ca9/Screenshot_20191221-185419_PolicyLibDemo.jpg -------------------------------------------------------------------------------- /Screenshot_20191230-201024_PolicyLibDemo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jaychou2012/PolicyLib/d894e4ce3ffbe82b06624c8d731df0833ef51ca9/Screenshot_20191230-201024_PolicyLibDemo.jpg -------------------------------------------------------------------------------- /app-debug.apk: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jaychou2012/PolicyLib/d894e4ce3ffbe82b06624c8d731df0833ef51ca9/app-debug.apk -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 28 5 | defaultConfig { 6 | applicationId "com.db.policylibdemo" 7 | minSdkVersion 15 8 | targetSdkVersion 28 9 | multiDexEnabled true 10 | versionCode 1 11 | versionName "1.0" 12 | testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" 13 | } 14 | buildTypes { 15 | release { 16 | debuggable false 17 | minifyEnabled false 18 | proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' 19 | } 20 | } 21 | 22 | compileOptions { 23 | sourceCompatibility JavaVersion.VERSION_1_8 24 | targetCompatibility JavaVersion.VERSION_1_8 25 | } 26 | } 27 | 28 | dependencies { 29 | implementation fileTree(dir: 'libs', include: ['*.jar']) 30 | implementation 'androidx.appcompat:appcompat:1.0.2' 31 | implementation 'androidx.constraintlayout:constraintlayout:1.1.3' 32 | implementation project(':policylib') 33 | testImplementation 'junit:junit:4.12' 34 | androidTestImplementation 'androidx.test.ext:junit:1.1.0' 35 | androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1' 36 | } 37 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # You can control the set of applied configuration files using the 3 | # proguardFiles setting in build.gradle. 4 | # 5 | # For more details, see 6 | # http://developer.android.com/guide/developing/tools/proguard.html 7 | 8 | # If your project uses WebView with JS, uncomment the following 9 | # and specify the fully qualified class name to the JavaScript interface 10 | # class: 11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 | # public *; 13 | #} 14 | 15 | # Uncomment this to preserve the line number information for 16 | # debugging stack traces. 17 | #-keepattributes SourceFile,LineNumberTable 18 | 19 | # If you keep the line number information, uncomment this to 20 | # hide the original source file name. 21 | #-renamesourcefileattribute SourceFile 22 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/db/policylibdemo/ExampleInstrumentedTest.java: -------------------------------------------------------------------------------- 1 | package com.db.policylibdemo; 2 | 3 | import android.content.Context; 4 | 5 | import androidx.test.platform.app.InstrumentationRegistry; 6 | import androidx.test.ext.junit.runners.AndroidJUnit4; 7 | 8 | import org.junit.Test; 9 | import org.junit.runner.RunWith; 10 | 11 | import static org.junit.Assert.*; 12 | 13 | /** 14 | * Instrumented test, which will execute on an Android device. 15 | * 16 | * @see Testing documentation 17 | */ 18 | @RunWith(AndroidJUnit4.class) 19 | public class ExampleInstrumentedTest { 20 | @Test 21 | public void useAppContext() { 22 | // Context of the app under test. 23 | Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext(); 24 | 25 | assertEquals("com.db.policylibdemo", appContext.getPackageName()); 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 20 | 21 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | 50 | 51 | 52 | -------------------------------------------------------------------------------- /app/src/main/java/com/db/policylibdemo/BaseApplication.java: -------------------------------------------------------------------------------- 1 | package com.db.policylibdemo; 2 | 3 | import android.app.Application; 4 | import android.content.Context; 5 | 6 | import androidx.multidex.MultiDex; 7 | 8 | public class BaseApplication extends Application { 9 | @Override 10 | protected void attachBaseContext(Context base) { 11 | super.attachBaseContext(base); 12 | MultiDex.install(base); 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /app/src/main/java/com/db/policylibdemo/MainActivity.java: -------------------------------------------------------------------------------- 1 | package com.db.policylibdemo; 2 | 3 | import android.Manifest; 4 | import android.content.Intent; 5 | import android.os.Bundle; 6 | import android.view.View; 7 | import android.widget.Button; 8 | import android.widget.TextView; 9 | import android.widget.Toast; 10 | 11 | import androidx.annotation.NonNull; 12 | import androidx.annotation.Nullable; 13 | import androidx.appcompat.app.AppCompatActivity; 14 | 15 | import com.db.policylib.PermissionPolicy; 16 | import com.db.policylib.Policy; 17 | import com.db.policylib.UpdateUtils; 18 | 19 | import java.util.ArrayList; 20 | import java.util.List; 21 | 22 | import pub.devrel.easypermissions.AfterPermissionGranted; 23 | import pub.devrel.easypermissions.AppSettingsDialog; 24 | import pub.devrel.easypermissions.EasyPermissions; 25 | 26 | public class MainActivity extends AppCompatActivity implements Policy.RuleListener, EasyPermissions.PermissionCallbacks, 27 | EasyPermissions.RationaleCallbacks, Policy.PolicyClick, View.OnClickListener { 28 | private TextView tv_text; 29 | private Button btn_update; 30 | private List list; 31 | private static final String[] STORAGE_AND_PHONE = 32 | {Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.READ_PHONE_STATE}; 33 | private static final int RC_STORAGE_PHONE_PERM = 125; 34 | private MainFragment mainFragment; 35 | private String text = "欢迎使用XX应用!我们将通过XXXXXX《用户协议》和《隐私政策》帮助您了解我们收集、使用、存储和共享个人信息的情况,以及您所享有的相关权利。\n\n" + 36 | "• 为了向您提供XX音频文件生成存储、头像上传、用户注册等功能服务,我们需要使用您的一些存储权限、音视频录制权限、相机权限、获取设备信息等权限及信息。\n" + 37 | "• 您可以在个人中心修改、更正您的信息,也可以自己注销账户。\n" + 38 | "• 我们会采用业界领先的安全技术保护好您的个人信息。\n\n" + 39 | "您可以通过阅读完整版用户隐私政策,了解个人信息类型与用途的对应关系等更加详尽的个人信息处理规则。\n" + 40 | "如您同意,请点击“同意”开始接受我们的服务。"; 41 | private String content = "新版本更新如下内容:\n1、优化界面;\n2、适配9.0系统;\n3、其他优化"; 42 | private String url = "http://gdown.baidu.com/data/wisegame/7206f8fe2dc6b0ed/QQyouxiang_10142253.apk"; 43 | 44 | @Override 45 | protected void onCreate(Bundle savedInstanceState) { 46 | super.onCreate(savedInstanceState); 47 | setContentView(R.layout.activity_main); 48 | initRule(); 49 | } 50 | 51 | private void initRule() { 52 | //展示用户协议和隐私政策 53 | Policy.getInstance().showRuleDialog(this, "用户协议和隐私政策概要", text, R.color.link, this); 54 | } 55 | 56 | @Override 57 | public void rule(boolean agree) { 58 | if (agree) { 59 | showBeforePolicyDialog(); 60 | } else { 61 | MainActivity.this.finish(); 62 | } 63 | } 64 | 65 | @Override 66 | public void oneClick() { 67 | Intent intent = new Intent(this, RuleActivity.class); 68 | intent.putExtra("privateRule", false); 69 | intent.putExtra("url", "file:////android_asset/userRule.html"); 70 | startActivity(intent); 71 | } 72 | 73 | @Override 74 | public void twoClick() { 75 | Intent intent = new Intent(this, RuleActivity.class); 76 | intent.putExtra("privateRule", true); 77 | intent.putExtra("url", "file:////android_asset/privateRule.html"); 78 | startActivity(intent); 79 | } 80 | 81 | private void initView() { 82 | tv_text = findViewById(R.id.tv_text); 83 | btn_update = findViewById(R.id.btn_update); 84 | mainFragment = (MainFragment) getSupportFragmentManager().findFragmentById(R.id.fragment); 85 | tv_text.setText("必要权限已经可以使用"); 86 | btn_update.setOnClickListener(this); 87 | } 88 | 89 | private void showBeforePolicyDialog() { 90 | list = new ArrayList<>(); 91 | PermissionPolicy permissionPolicy = new PermissionPolicy(); 92 | permissionPolicy.setPermission(Manifest.permission.READ_EXTERNAL_STORAGE); 93 | permissionPolicy.setTitle("存储权限"); 94 | permissionPolicy.setDes("缓存图片和视频,降低流量消耗。"); 95 | permissionPolicy.setIcon(R.mipmap.icon_storage); 96 | permissionPolicy.setRequest(true); 97 | 98 | PermissionPolicy permissionPolicy1 = new PermissionPolicy(); 99 | permissionPolicy1.setPermission(Manifest.permission.READ_PHONE_STATE); 100 | permissionPolicy1.setTitle("手机/电话权限"); 101 | permissionPolicy1.setDes("校验IMEI&IMSI码,防止账号被盗。"); 102 | permissionPolicy1.setIcon(R.mipmap.icon_tel); 103 | permissionPolicy1.setRequest(false); 104 | if (!Policy.getInstance().hasPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE)) { 105 | list.add(permissionPolicy); 106 | } 107 | if (!Policy.getInstance().hasPermission(this, Manifest.permission.READ_PHONE_STATE)) { 108 | list.add(permissionPolicy1); 109 | } 110 | if (list.size() == 0) { 111 | initView(); 112 | return; 113 | } 114 | getPermission(); 115 | } 116 | 117 | @AfterPermissionGranted(RC_STORAGE_PHONE_PERM) 118 | public void getPermission() { 119 | if (EasyPermissions.hasPermissions(this, STORAGE_AND_PHONE)) { 120 | initView(); 121 | } else { 122 | EasyPermissions.requestPermissions( 123 | MainActivity.this, "权限", 124 | RC_STORAGE_PHONE_PERM, list, 125 | STORAGE_AND_PHONE); 126 | } 127 | } 128 | 129 | @Override 130 | public void onRequestPermissionsResult(int requestCode, 131 | @NonNull String[] permissions, 132 | @NonNull int[] grantResults) { 133 | super.onRequestPermissionsResult(requestCode, permissions, grantResults); 134 | if (requestCode == RC_STORAGE_PHONE_PERM) { 135 | EasyPermissions.onRequestPermissionsResult(requestCode, permissions, grantResults, this); 136 | } else if (requestCode == MainFragment.RC_RECORD_AUDIO_PERM) { 137 | EasyPermissions.onRequestPermissionsResult(requestCode, permissions, grantResults, mainFragment); 138 | } 139 | } 140 | 141 | @Override 142 | protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { 143 | super.onActivityResult(requestCode, resultCode, data); 144 | if (requestCode == RC_STORAGE_PHONE_PERM) { 145 | getPermission(); 146 | } else if (requestCode == MainFragment.RC_RECORD_AUDIO_PERM) { 147 | mainFragment.onActivityResult(requestCode, resultCode, data); 148 | } 149 | } 150 | 151 | @Override 152 | public void onPermissionsGranted(int requestCode, @NonNull List perms) { 153 | 154 | } 155 | 156 | @Override 157 | public void onPermissionsDenied(int requestCode, @NonNull List perms) { 158 | if (EasyPermissions.somePermissionPermanentlyDenied(this, perms)) { 159 | new AppSettingsDialog.Builder(this).build().show(requestCode, list, this); 160 | } else { 161 | getPermission(); 162 | } 163 | } 164 | 165 | @Override 166 | public void onRationaleAccepted(int requestCode) { 167 | 168 | } 169 | 170 | @Override 171 | public void onRationaleDenied(int requestCode) { 172 | showToast("必要的权限被禁止,无法正常使用"); 173 | } 174 | 175 | private void showToast(String text) { 176 | Toast.makeText(this, text, Toast.LENGTH_SHORT).show(); 177 | } 178 | 179 | @Override 180 | public void policyCancelClick(int reqeustCode) { 181 | showToast("必要的授权权限被禁止,无法正常使用"); 182 | } 183 | 184 | @Override 185 | public void onClick(View v) { 186 | switch (v.getId()) { 187 | case R.id.btn_update: 188 | UpdateUtils.getInstance().showUpdate(this, content, url, false); 189 | break; 190 | } 191 | } 192 | 193 | } 194 | -------------------------------------------------------------------------------- /app/src/main/java/com/db/policylibdemo/MainFragment.java: -------------------------------------------------------------------------------- 1 | package com.db.policylibdemo; 2 | 3 | import android.Manifest; 4 | import android.content.Intent; 5 | import android.os.Bundle; 6 | import android.view.LayoutInflater; 7 | import android.view.View; 8 | import android.view.ViewGroup; 9 | import android.widget.Button; 10 | import android.widget.Toast; 11 | 12 | import androidx.annotation.NonNull; 13 | import androidx.annotation.Nullable; 14 | import androidx.fragment.app.Fragment; 15 | 16 | import com.db.policylib.PermissionPolicy; 17 | import com.db.policylib.Policy; 18 | 19 | import java.util.ArrayList; 20 | import java.util.List; 21 | 22 | import pub.devrel.easypermissions.AfterPermissionGranted; 23 | import pub.devrel.easypermissions.AppSettingsDialog; 24 | import pub.devrel.easypermissions.EasyPermissions; 25 | 26 | 27 | public class MainFragment extends Fragment implements View.OnClickListener, EasyPermissions.PermissionCallbacks, 28 | EasyPermissions.RationaleCallbacks, Policy.PolicyClick { 29 | public static final int RC_RECORD_AUDIO_PERM = 122; 30 | private static final String[] RECORD_AUDIO = 31 | {Manifest.permission.RECORD_AUDIO}; 32 | private List list; 33 | private Button btn_audio; 34 | 35 | public MainFragment() { 36 | // Required empty public constructor 37 | } 38 | 39 | @Override 40 | public View onCreateView(LayoutInflater inflater, ViewGroup container, 41 | Bundle savedInstanceState) { 42 | 43 | return inflater.inflate(R.layout.fragment_main, container, false); 44 | } 45 | 46 | @Override 47 | public void onActivityCreated(@Nullable Bundle savedInstanceState) { 48 | super.onActivityCreated(savedInstanceState); 49 | btn_audio = getView().findViewById(R.id.btn_audio); 50 | btn_audio.setOnClickListener(this); 51 | list = new ArrayList<>(); 52 | PermissionPolicy permissionPolicy = new PermissionPolicy(); 53 | permissionPolicy.setPermission(Manifest.permission.RECORD_AUDIO); 54 | permissionPolicy.setTitle("录音权限"); 55 | permissionPolicy.setDes("用于录制音频功能。"); 56 | permissionPolicy.setIcon(R.mipmap.icon_record_audio); 57 | permissionPolicy.setRequest(true); 58 | list.add(permissionPolicy); 59 | } 60 | 61 | @Override 62 | public void onActivityResult(int requestCode, int resultCode, Intent data) { 63 | super.onActivityResult(requestCode, resultCode, data); 64 | if (requestCode == RC_RECORD_AUDIO_PERM) { 65 | showToast("设置回调"); 66 | getPermissions(); 67 | } 68 | } 69 | 70 | @Override 71 | public void onRequestPermissionsResult(int requestCode, 72 | @NonNull String[] permissions, 73 | @NonNull int[] grantResults) { 74 | super.onRequestPermissionsResult(requestCode, permissions, grantResults); 75 | // EasyPermissions handles the request result. 76 | EasyPermissions.onRequestPermissionsResult(requestCode, permissions, grantResults, this); 77 | } 78 | 79 | @AfterPermissionGranted(RC_RECORD_AUDIO_PERM) 80 | private void getPermissions() { 81 | if (EasyPermissions.hasPermissions(getContext(), RECORD_AUDIO)) { 82 | // Have permission, do the thing! 83 | showToast("已获取权限"); 84 | } else { 85 | // Request one permission 86 | EasyPermissions.requestPermissions(this, "权限", 87 | RC_RECORD_AUDIO_PERM, list, RECORD_AUDIO); 88 | } 89 | } 90 | 91 | @Override 92 | public void onPermissionsGranted(int requestCode, @NonNull List perms) { 93 | 94 | } 95 | 96 | @Override 97 | public void onPermissionsDenied(int requestCode, @NonNull List perms) { 98 | if (EasyPermissions.somePermissionPermanentlyDenied(this, perms)) { 99 | new AppSettingsDialog.Builder(this).build().show(requestCode, list, this); 100 | } else { 101 | getPermissions(); 102 | } 103 | } 104 | 105 | @Override 106 | public void onClick(View v) { 107 | switch (v.getId()) { 108 | case R.id.btn_audio: 109 | getPermissions(); 110 | break; 111 | } 112 | } 113 | 114 | @Override 115 | public void onRationaleAccepted(int requestCode) { 116 | 117 | } 118 | 119 | @Override 120 | public void onRationaleDenied(int requestCode) { 121 | showToast("取消授权"); 122 | } 123 | 124 | @Override 125 | public void policyCancelClick(int reqeustCode) { 126 | showToast("必要的授权权限被禁止,无法正常使用"); 127 | } 128 | 129 | private void showToast(String text) { 130 | Toast.makeText(getActivity(), text, Toast.LENGTH_SHORT).show(); 131 | } 132 | } 133 | -------------------------------------------------------------------------------- /app/src/main/java/com/db/policylibdemo/RuleActivity.java: -------------------------------------------------------------------------------- 1 | package com.db.policylibdemo; 2 | 3 | import android.graphics.Bitmap; 4 | import android.media.MediaPlayer; 5 | import android.net.http.SslError; 6 | import android.os.Build; 7 | import android.os.Bundle; 8 | import android.view.View; 9 | import android.webkit.SslErrorHandler; 10 | import android.webkit.WebChromeClient; 11 | import android.webkit.WebResourceError; 12 | import android.webkit.WebResourceRequest; 13 | import android.webkit.WebSettings; 14 | import android.webkit.WebView; 15 | import android.webkit.WebViewClient; 16 | import android.widget.Toast; 17 | 18 | import androidx.appcompat.app.AppCompatActivity; 19 | import androidx.appcompat.widget.Toolbar; 20 | 21 | public class RuleActivity extends AppCompatActivity { 22 | private Toolbar toolBar; 23 | private WebView webView; 24 | private WebSettings webSettings; 25 | private String url; 26 | private boolean priveteRule; 27 | 28 | @Override 29 | protected void onCreate(Bundle savedInstanceState) { 30 | super.onCreate(savedInstanceState); 31 | setContentView(R.layout.activity_rule); 32 | initView(); 33 | } 34 | 35 | private void initView() { 36 | url = getIntent().getStringExtra("url"); 37 | priveteRule = getIntent().getBooleanExtra("privateRule", true); 38 | toolBar = findViewById(R.id.toolBar); 39 | webView = findViewById(R.id.webview); 40 | webSettings = webView.getSettings(); 41 | webSettings.setJavaScriptCanOpenWindowsAutomatically(true); 42 | webSettings.setJavaScriptEnabled(true); 43 | if (Build.VERSION.SDK_INT == 18) { 44 | webView.removeJavascriptInterface("searchBoxJavaBridge_"); 45 | webView.removeJavascriptInterface("accessibility"); 46 | webView.removeJavascriptInterface("accessibilityTraversal"); 47 | } 48 | webSettings = webView.getSettings(); 49 | webSettings.setJavaScriptEnabled(true); 50 | webSettings.setJavaScriptCanOpenWindowsAutomatically(true); 51 | webSettings.setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN); 52 | webSettings.setUseWideViewPort(true); 53 | webSettings.setLoadWithOverviewMode(true); 54 | webSettings.setCacheMode(WebSettings.LOAD_DEFAULT); 55 | webSettings.setDatabaseEnabled(true); 56 | webSettings.setDomStorageEnabled(true); 57 | webSettings.setAppCacheEnabled(true); 58 | webSettings.setSupportZoom(true); 59 | webSettings.setBuiltInZoomControls(true); 60 | webSettings.setDisplayZoomControls(false); 61 | webSettings.setSavePassword(false); 62 | webSettings.setSaveFormData(false); 63 | if (Build.VERSION.SDK_INT >= 19) { 64 | webSettings.setLoadsImagesAutomatically(true); 65 | } else { 66 | webSettings.setLoadsImagesAutomatically(false); 67 | } 68 | webView.setWebViewClient(new RuleWebViewClient()); 69 | webView.setWebChromeClient(new RuleWebChromeClient()); 70 | webView.loadUrl(url); 71 | toolBar.setTitle(priveteRule ? "隐私政策" : "用户协议"); 72 | toolBar.setNavigationIcon(R.mipmap.icon_back_arrow); 73 | setSupportActionBar(toolBar); 74 | toolBar.setNavigationOnClickListener(new View.OnClickListener() { 75 | @Override 76 | public void onClick(View v) { 77 | finish(); 78 | } 79 | }); 80 | } 81 | 82 | private class RuleWebChromeClient extends WebChromeClient { 83 | @Override 84 | public void onProgressChanged(WebView view, int newProgress) { 85 | super.onProgressChanged(view, newProgress); 86 | } 87 | 88 | private class VideoCompletionListener implements MediaPlayer.OnCompletionListener, 89 | MediaPlayer.OnErrorListener { 90 | 91 | @Override 92 | public boolean onError(MediaPlayer mp, int what, int extra) { 93 | return false; 94 | } 95 | 96 | @Override 97 | public void onCompletion(MediaPlayer mp) { 98 | onHideCustomView(); 99 | } 100 | 101 | } 102 | } 103 | 104 | private class RuleWebViewClient extends WebViewClient { 105 | 106 | @Override 107 | public boolean shouldOverrideUrlLoading(WebView webView, String s) { 108 | webView.loadUrl(s); 109 | return true; 110 | } 111 | 112 | @Override 113 | public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) { 114 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 115 | view.loadUrl(request.getUrl().toString()); 116 | } else { 117 | view.loadUrl(request.toString()); 118 | } 119 | return true; 120 | } 121 | 122 | @Override 123 | public void onLoadResource(WebView view, String url) { 124 | super.onLoadResource(view, url); 125 | } 126 | 127 | @Override 128 | public void onPageStarted(WebView view, String url, Bitmap favicon) { 129 | super.onPageStarted(view, url, favicon); 130 | } 131 | 132 | @Override 133 | public void onPageFinished(WebView view, String url) { 134 | super.onPageFinished(view, url); 135 | if (!webSettings.getLoadsImagesAutomatically()) { 136 | webSettings.setLoadsImagesAutomatically(true); 137 | } 138 | } 139 | 140 | @Override 141 | public void onReceivedError(WebView view, WebResourceRequest request, WebResourceError error) { 142 | super.onReceivedError(view, request, error); 143 | showToast("加载出错"); 144 | } 145 | 146 | @Override 147 | public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) { 148 | handler.cancel(); 149 | } 150 | } 151 | 152 | private void showToast(String text) { 153 | Toast.makeText(this, text, Toast.LENGTH_SHORT).show(); 154 | } 155 | } 156 | -------------------------------------------------------------------------------- /app/src/main/res/drawable-v24/ic_launcher_foreground.xml: -------------------------------------------------------------------------------- 1 | 7 | 12 | 13 | 19 | 22 | 25 | 26 | 27 | 28 | 34 | 35 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_launcher_background.xml: -------------------------------------------------------------------------------- 1 | 2 | 7 | 10 | 15 | 20 | 25 | 30 | 35 | 40 | 45 | 50 | 55 | 60 | 65 | 70 | 75 | 80 | 85 | 90 | 95 | 100 | 105 | 110 | 115 | 120 | 125 | 130 | 135 | 140 | 145 | 150 | 155 | 160 | 165 | 170 | 171 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 9 | 10 | 16 | 17 |