@ -0,0 +1,10 @@ |
|||||||
|
*.iml |
||||||
|
.gradle |
||||||
|
/local.properties |
||||||
|
/.idea/libraries |
||||||
|
/.idea/modules.xml |
||||||
|
/.idea/workspace.xml |
||||||
|
.DS_Store |
||||||
|
/build |
||||||
|
/captures |
||||||
|
.externalNativeBuild |
@ -0,0 +1,57 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<project version="4"> |
||||||
|
<component name="WizardSettings"> |
||||||
|
<option name="children"> |
||||||
|
<map> |
||||||
|
<entry key="imageWizard"> |
||||||
|
<value> |
||||||
|
<PersistentState> |
||||||
|
<option name="children"> |
||||||
|
<map> |
||||||
|
<entry key="imageAssetPanel"> |
||||||
|
<value> |
||||||
|
<PersistentState> |
||||||
|
<option name="children"> |
||||||
|
<map> |
||||||
|
<entry key="launcher"> |
||||||
|
<value> |
||||||
|
<PersistentState> |
||||||
|
<option name="children"> |
||||||
|
<map> |
||||||
|
<entry key="foregroundImage"> |
||||||
|
<value> |
||||||
|
<PersistentState> |
||||||
|
<option name="values"> |
||||||
|
<map> |
||||||
|
<entry key="scalingPercent" value="50" /> |
||||||
|
</map> |
||||||
|
</option> |
||||||
|
</PersistentState> |
||||||
|
</value> |
||||||
|
</entry> |
||||||
|
</map> |
||||||
|
</option> |
||||||
|
<option name="values"> |
||||||
|
<map> |
||||||
|
<entry key="backgroundAssetType" value="COLOR" /> |
||||||
|
<entry key="backgroundColor" value="ffffff" /> |
||||||
|
<entry key="foregroundImage" value="C:\app_updater.png" /> |
||||||
|
</map> |
||||||
|
</option> |
||||||
|
</PersistentState> |
||||||
|
</value> |
||||||
|
</entry> |
||||||
|
</map> |
||||||
|
</option> |
||||||
|
</PersistentState> |
||||||
|
</value> |
||||||
|
</entry> |
||||||
|
</map> |
||||||
|
</option> |
||||||
|
</PersistentState> |
||||||
|
</value> |
||||||
|
</entry> |
||||||
|
</map> |
||||||
|
</option> |
||||||
|
</component> |
||||||
|
</project> |
@ -0,0 +1,29 @@ |
|||||||
|
<component name="ProjectCodeStyleConfiguration"> |
||||||
|
<code_scheme name="Project" version="173"> |
||||||
|
<Objective-C-extensions> |
||||||
|
<file> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Import" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Macro" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Typedef" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Enum" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Constant" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Global" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Struct" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="FunctionPredecl" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Function" /> |
||||||
|
</file> |
||||||
|
<class> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Property" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Synthesize" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="InitMethod" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="StaticMethod" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="InstanceMethod" /> |
||||||
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="DeallocMethod" /> |
||||||
|
</class> |
||||||
|
<extensions> |
||||||
|
<pair source="cpp" header="h" fileNamingConvention="NONE" /> |
||||||
|
<pair source="c" header="h" fileNamingConvention="NONE" /> |
||||||
|
</extensions> |
||||||
|
</Objective-C-extensions> |
||||||
|
</code_scheme> |
||||||
|
</component> |
@ -0,0 +1,6 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<project version="4"> |
||||||
|
<component name="Encoding" defaultCharsetForPropertiesFiles="UTF-8"> |
||||||
|
<file url="PROJECT" charset="UTF-8" /> |
||||||
|
</component> |
||||||
|
</project> |
@ -0,0 +1,20 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<project version="4"> |
||||||
|
<component name="GradleSettings"> |
||||||
|
<option name="linkedExternalProjectsSettings"> |
||||||
|
<GradleProjectSettings> |
||||||
|
<option name="distributionType" value="DEFAULT_WRAPPED" /> |
||||||
|
<option name="externalProjectPath" value="$PROJECT_DIR$" /> |
||||||
|
<option name="modules"> |
||||||
|
<set> |
||||||
|
<option value="$PROJECT_DIR$" /> |
||||||
|
<option value="$PROJECT_DIR$/app" /> |
||||||
|
<option value="$PROJECT_DIR$/app-dialog" /> |
||||||
|
<option value="$PROJECT_DIR$/app-updater" /> |
||||||
|
</set> |
||||||
|
</option> |
||||||
|
<option name="resolveModulePerSourceSet" value="false" /> |
||||||
|
</GradleProjectSettings> |
||||||
|
</option> |
||||||
|
</component> |
||||||
|
</project> |
@ -0,0 +1,34 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<project version="4"> |
||||||
|
<component name="NullableNotNullManager"> |
||||||
|
<option name="myDefaultNullable" value="android.support.annotation.Nullable" /> |
||||||
|
<option name="myDefaultNotNull" value="android.support.annotation.NonNull" /> |
||||||
|
<option name="myNullables"> |
||||||
|
<value> |
||||||
|
<list size="5"> |
||||||
|
<item index="0" class="java.lang.String" itemvalue="org.jetbrains.annotations.Nullable" /> |
||||||
|
<item index="1" class="java.lang.String" itemvalue="javax.annotation.Nullable" /> |
||||||
|
<item index="2" class="java.lang.String" itemvalue="javax.annotation.CheckForNull" /> |
||||||
|
<item index="3" class="java.lang.String" itemvalue="edu.umd.cs.findbugs.annotations.Nullable" /> |
||||||
|
<item index="4" class="java.lang.String" itemvalue="android.support.annotation.Nullable" /> |
||||||
|
</list> |
||||||
|
</value> |
||||||
|
</option> |
||||||
|
<option name="myNotNulls"> |
||||||
|
<value> |
||||||
|
<list size="4"> |
||||||
|
<item index="0" class="java.lang.String" itemvalue="org.jetbrains.annotations.NotNull" /> |
||||||
|
<item index="1" class="java.lang.String" itemvalue="javax.annotation.Nonnull" /> |
||||||
|
<item index="2" class="java.lang.String" itemvalue="edu.umd.cs.findbugs.annotations.NonNull" /> |
||||||
|
<item index="3" class="java.lang.String" itemvalue="android.support.annotation.NonNull" /> |
||||||
|
</list> |
||||||
|
</value> |
||||||
|
</option> |
||||||
|
</component> |
||||||
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_7" project-jdk-name="1.8" project-jdk-type="JavaSDK"> |
||||||
|
<output url="file://$PROJECT_DIR$/build/classes" /> |
||||||
|
</component> |
||||||
|
<component name="ProjectType"> |
||||||
|
<option name="id" value="Android" /> |
||||||
|
</component> |
||||||
|
</project> |
@ -0,0 +1,12 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<project version="4"> |
||||||
|
<component name="RunConfigurationProducerService"> |
||||||
|
<option name="ignoredProducers"> |
||||||
|
<set> |
||||||
|
<option value="org.jetbrains.plugins.gradle.execution.test.runner.AllInPackageGradleConfigurationProducer" /> |
||||||
|
<option value="org.jetbrains.plugins.gradle.execution.test.runner.TestClassGradleConfigurationProducer" /> |
||||||
|
<option value="org.jetbrains.plugins.gradle.execution.test.runner.TestMethodGradleConfigurationProducer" /> |
||||||
|
</set> |
||||||
|
</option> |
||||||
|
</component> |
||||||
|
</project> |
@ -0,0 +1,6 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<project version="4"> |
||||||
|
<component name="VcsDirectoryMappings"> |
||||||
|
<mapping directory="$PROJECT_DIR$" vcs="Git" /> |
||||||
|
</component> |
||||||
|
</project> |
@ -0,0 +1,23 @@ |
|||||||
|
|
||||||
|
The MIT License (MIT) |
||||||
|
Copyright (c) 2017 Jenly Yu |
||||||
|
https://github.com/jenly1314 |
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining |
||||||
|
a copy of this software and associated documentation files |
||||||
|
(the "Software"), to deal in the Software without restriction, including |
||||||
|
without limitation the rights to use, copy, modify, merge, publish, |
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to permit |
||||||
|
persons to whom the Software is furnished to do so, subject to the |
||||||
|
following conditions: |
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included |
||||||
|
in all copies or substantial portions of the Software. |
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
||||||
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
||||||
|
DEALINGS IN THE SOFTWARE. |
@ -0,0 +1,142 @@ |
|||||||
|
# AppUpdater |
||||||
|
|
||||||
|
![Image](app/ic_launcher-web.png) |
||||||
|
|
||||||
|
[![Download](https://img.shields.io/badge/download-App-blue.svg)](https://raw.githubusercontent.com/jenly1314/AppUpdater/master/app/release/app-release.apk) |
||||||
|
[![](https://jitpack.io/v/jenly1314/AppUpdater.svg)](https://jitpack.io/#jenly1314/AppUpdater) |
||||||
|
[![License](https://img.shields.io/badge/license-MIT-blue.svg)](https://opensource.org/licenses/mit-license.php) |
||||||
|
[![Blog](https://img.shields.io/badge/blog-Jenly-9933CC.svg)](http://blog.csdn.net/jenly121) |
||||||
|
|
||||||
|
AppUpdater for Android 是一个专注于App更新一键傻瓜式集成的开源库,主要包括app-updater和app-dialog。 |
||||||
|
> 下载更新和弹框提示分开,是因为这本来就是两个逻辑。完全独立开来能有效的解耦。 |
||||||
|
* app-updater 主要负责后台下载更新App,无需担心下载时各种配置相关的细节,一键傻瓜式升级。 |
||||||
|
* app-dialog 主要是提供常用的Dialog和DialogFragment,简化弹框提示,样式支持高度自定义。 |
||||||
|
> app-updater + app-dialog 配合使用,谁用谁知道。 |
||||||
|
|
||||||
|
|
||||||
|
## 功能介绍 |
||||||
|
- [x] 专注于App更新一键傻瓜式升级 |
||||||
|
- [x] 支持下载监听 |
||||||
|
- [x] 支持下载失败,重新下载 |
||||||
|
- [x] 支持通知栏提示内容和过程全部可配置 |
||||||
|
- [x] 支持Android O |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Gif 展示 |
||||||
|
![Image](GIF.gif) |
||||||
|
|
||||||
|
## 引入 |
||||||
|
|
||||||
|
### Maven: |
||||||
|
```maven |
||||||
|
//app-updater |
||||||
|
<dependency> |
||||||
|
<groupId>com.king.app</groupId> |
||||||
|
<artifactId>app-updater</artifactId> |
||||||
|
<version>1.0</version> |
||||||
|
<type>pom</type> |
||||||
|
</dependency> |
||||||
|
|
||||||
|
//app-dialog |
||||||
|
<dependency> |
||||||
|
<groupId>com.king.app</groupId> |
||||||
|
<artifactId>app-dialog</artifactId> |
||||||
|
<version>1.0</version> |
||||||
|
<type>pom</type> |
||||||
|
</dependency> |
||||||
|
``` |
||||||
|
### Gradle: |
||||||
|
```gradle |
||||||
|
//app-updater |
||||||
|
compile 'com.king.app:app-updater:1.0' |
||||||
|
|
||||||
|
//app-dialog |
||||||
|
compile 'com.king.app:app-dialog:1.0' |
||||||
|
``` |
||||||
|
### Lvy: |
||||||
|
```lvy |
||||||
|
//app-updater |
||||||
|
<dependency org='com.king.app' name='app-dialog' rev='1.0'> |
||||||
|
<artifact name='$AID' ext='pom'></artifact> |
||||||
|
</dependency> |
||||||
|
|
||||||
|
//app-dialog |
||||||
|
<dependency org='com.king.app' name='app-dialog' rev='1.0'> |
||||||
|
<artifact name='$AID' ext='pom'></artifact> |
||||||
|
</dependency> |
||||||
|
``` |
||||||
|
|
||||||
|
###### 如果Gradle出现compile失败的情况,可以在Project的build.gradle里面添加如下:(也可以使用上面的GitPack来complie) |
||||||
|
```gradle |
||||||
|
allprojects { |
||||||
|
repositories { |
||||||
|
//... |
||||||
|
maven { url 'https://dl.bintray.com/jenly/maven' } |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
## 示例 |
||||||
|
|
||||||
|
```Java |
||||||
|
//一句代码,傻瓜式更新 |
||||||
|
new AppUpdater(getContext(),url).start(); |
||||||
|
``` |
||||||
|
```Java |
||||||
|
//简单弹框升级 |
||||||
|
AppDialogConfig config = new AppDialogConfig(); |
||||||
|
config.setTitle("简单弹框升级") |
||||||
|
.setOk("升级") |
||||||
|
.setContent("1、新增某某功能、\n2、修改某某问题、\n3、优化某某BUG、") |
||||||
|
.setOnClickOk(new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
new AppUpdater.Builder() |
||||||
|
.serUrl(mUrl) |
||||||
|
.setFilename("AppUpdater.apk") |
||||||
|
.build(getContext()) |
||||||
|
.start(); |
||||||
|
AppDialog.INSTANCE.dismissDialog(); |
||||||
|
} |
||||||
|
}); |
||||||
|
AppDialog.INSTANCE.showDialog(getContext(),config); |
||||||
|
``` |
||||||
|
```Java |
||||||
|
//简单DialogFragment升级 |
||||||
|
AppDialogConfig config = new AppDialogConfig(); |
||||||
|
config.setTitle("简单DialogFragment升级") |
||||||
|
.setOk("升级") |
||||||
|
.setContent("1、新增某某功能、\n2、修改某某问题、\n3、优化某某BUG、") |
||||||
|
.setOnClickOk(new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
new AppUpdater.Builder() |
||||||
|
.serUrl(mUrl) |
||||||
|
.setFilename("AppUpdater.apk") |
||||||
|
.build(getContext()) |
||||||
|
.start(); |
||||||
|
AppDialog.INSTANCE.dismissDialogFragment(getSupportFragmentManager()); |
||||||
|
} |
||||||
|
}); |
||||||
|
AppDialog.INSTANCE.showDialogFragment(getSupportFragmentManager(),config); |
||||||
|
|
||||||
|
``` |
||||||
|
|
||||||
|
更多使用示例请查看[App](app)。 |
||||||
|
|
||||||
|
|
||||||
|
## 关于我 |
||||||
|
Name: <a title="关于作者" href="https://about.me/jenly1314" target="_blank">Jenly</a> |
||||||
|
|
||||||
|
Email: <a title="欢迎邮件与我交流" href="mailto:jenly1314@gmail.com" target="_blank">jenly1314#gmail.com</a> / <a title="给我发邮件" href="mailto:jenly1314@vip.qq.com" target="_blank">jenly1314#vip.qq.com</a> |
||||||
|
|
||||||
|
CSDN: <a title="CSDN博客" href="http://blog.csdn.net/jenly121" target="_blank">jenly121</a> |
||||||
|
|
||||||
|
Github: <a title="Github开源项目" href="https://github.com/jenly1314" target="_blank">jenly1314</a> |
||||||
|
|
||||||
|
微信公众号: |
||||||
|
|
||||||
|
![公众号](http://olambmg9j.bkt.clouddn.com/jenly666.jpg) |
||||||
|
|
||||||
|
加入QQ群: <a title="点击加入QQ群" href="http://shang.qq.com/wpa/qunwpa?idkey=8fcc6a2f88552ea44b1411582c94fd124f7bb3ec227e2a400dbbfaad3dc2f5ad" target="_blank">20867961</a> |
@ -0,0 +1 @@ |
|||||||
|
/build |
@ -0,0 +1,10 @@ |
|||||||
|
apply plugin: 'com.novoda.bintray-release' |
||||||
|
//添加 |
||||||
|
publish { |
||||||
|
userOrg = 'jenly'//bintray.com用户名 |
||||||
|
groupId = 'com.king.app'//jcenter上的路径 |
||||||
|
artifactId = 'app-dialog'//项目名称 |
||||||
|
publishVersion = '1.0'//版本号 |
||||||
|
desc = 'AppDialog for Android'//描述 |
||||||
|
website = 'https://github.com/jenly1314/AppUpdater'//网站 |
||||||
|
} |
@ -0,0 +1,42 @@ |
|||||||
|
apply plugin: 'com.android.library' |
||||||
|
apply from: 'bintray.gradle' |
||||||
|
|
||||||
|
android { |
||||||
|
compileSdkVersion 27 |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
defaultConfig { |
||||||
|
minSdkVersion 15 |
||||||
|
targetSdkVersion 27 |
||||||
|
versionCode 1 |
||||||
|
versionName "1.0" |
||||||
|
|
||||||
|
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
buildTypes { |
||||||
|
release { |
||||||
|
minifyEnabled false |
||||||
|
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
dependencies { |
||||||
|
implementation fileTree(dir: 'libs', include: ['*.jar']) |
||||||
|
|
||||||
|
compileOnly 'com.android.support:appcompat-v7:27.1.1' |
||||||
|
testImplementation 'junit:junit:4.12' |
||||||
|
androidTestImplementation 'com.android.support.test:runner:1.0.2' |
||||||
|
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2' |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -0,0 +1,21 @@ |
|||||||
|
# Add project specific ProGuard rules here. |
||||||
|
# You can control the set of applied configuration files using the |
||||||
|
# proguardFiles setting in build.gradle. |
||||||
|
# |
||||||
|
# For more details, see |
||||||
|
# http://developer.android.com/guide/developing/tools/proguard.html |
||||||
|
|
||||||
|
# If your project uses WebView with JS, uncomment the following |
||||||
|
# and specify the fully qualified class name to the JavaScript interface |
||||||
|
# class: |
||||||
|
#-keepclassmembers class fqcn.of.javascript.interface.for.webview { |
||||||
|
# public *; |
||||||
|
#} |
||||||
|
|
||||||
|
# Uncomment this to preserve the line number information for |
||||||
|
# debugging stack traces. |
||||||
|
#-keepattributes SourceFile,LineNumberTable |
||||||
|
|
||||||
|
# If you keep the line number information, uncomment this to |
||||||
|
# hide the original source file name. |
||||||
|
#-renamesourcefileattribute SourceFile |
@ -0,0 +1,26 @@ |
|||||||
|
package com.king.app.dialog; |
||||||
|
|
||||||
|
import android.content.Context; |
||||||
|
import android.support.test.InstrumentationRegistry; |
||||||
|
import android.support.test.runner.AndroidJUnit4; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
import org.junit.runner.RunWith; |
||||||
|
|
||||||
|
import static org.junit.Assert.*; |
||||||
|
|
||||||
|
/** |
||||||
|
* Instrumented test, which will execute on an Android device. |
||||||
|
* |
||||||
|
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a> |
||||||
|
*/ |
||||||
|
@RunWith(AndroidJUnit4.class) |
||||||
|
public class ExampleInstrumentedTest { |
||||||
|
@Test |
||||||
|
public void useAppContext() { |
||||||
|
// Context of the app under test.
|
||||||
|
Context appContext = InstrumentationRegistry.getTargetContext(); |
||||||
|
|
||||||
|
assertEquals("com.king.app.dialog.test", appContext.getPackageName()); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,2 @@ |
|||||||
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
package="com.king.app.dialog" /> |
@ -0,0 +1,194 @@ |
|||||||
|
package com.king.app.dialog; |
||||||
|
|
||||||
|
import android.app.Dialog; |
||||||
|
import android.content.Context; |
||||||
|
import android.content.DialogInterface; |
||||||
|
import android.support.annotation.StyleRes; |
||||||
|
import android.support.v4.app.DialogFragment; |
||||||
|
import android.support.v4.app.FragmentManager; |
||||||
|
import android.text.TextUtils; |
||||||
|
import android.view.KeyEvent; |
||||||
|
import android.view.LayoutInflater; |
||||||
|
import android.view.View; |
||||||
|
import android.view.Window; |
||||||
|
import android.view.WindowManager; |
||||||
|
import android.widget.Button; |
||||||
|
import android.widget.TextView; |
||||||
|
|
||||||
|
import com.king.app.dialog.fragment.AppDialogFragment; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public enum AppDialog { |
||||||
|
|
||||||
|
INSTANCE; |
||||||
|
|
||||||
|
private final float DEFAULT_WIDTH_RATIO = 0.85f; |
||||||
|
|
||||||
|
private Dialog mDialog; |
||||||
|
|
||||||
|
private String mTag; |
||||||
|
|
||||||
|
//-------------------------------------------
|
||||||
|
|
||||||
|
public View createAppDialogView(Context context,AppDialogConfig config){ |
||||||
|
View view = null; |
||||||
|
if(config!=null){ |
||||||
|
view = LayoutInflater.from(context).inflate(config.getLayoutId(),null); |
||||||
|
TextView tvDialogTitle = view.findViewById(config.getTitleId()); |
||||||
|
setText(tvDialogTitle,config.getTitle()); |
||||||
|
|
||||||
|
TextView tvDialogContent = view.findViewById(config.getContentId()); |
||||||
|
setText(tvDialogContent,config.getContent()); |
||||||
|
|
||||||
|
Button btnDialogCancel = view.findViewById(config.getCancelId()); |
||||||
|
setText(btnDialogCancel,config.getCancel()); |
||||||
|
btnDialogCancel.setOnClickListener(config.getOnClickCancel() != null ? config.getOnClickCancel() : mOnClickDismissDialog); |
||||||
|
btnDialogCancel.setVisibility(config.isHideCancel() ? View.GONE : View.VISIBLE); |
||||||
|
|
||||||
|
try{ |
||||||
|
//不强制要求要有横线
|
||||||
|
View line = view.findViewById(R.id.line); |
||||||
|
line.setVisibility(config.isHideCancel() ? View.GONE : View.VISIBLE); |
||||||
|
}catch (Exception e){ |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
Button btnDialogOK = view.findViewById(config.getOkId()); |
||||||
|
setText(btnDialogOK,config.getOk()); |
||||||
|
btnDialogOK.setOnClickListener(config.getOnClickOk() != null ? config.getOnClickOk() : mOnClickDismissDialog); |
||||||
|
} |
||||||
|
|
||||||
|
return view; |
||||||
|
} |
||||||
|
|
||||||
|
//-------------------------------------------
|
||||||
|
|
||||||
|
private View.OnClickListener mOnClickDismissDialog = new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
dismissDialog(); |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
private void setText(TextView tv,CharSequence text){ |
||||||
|
if(!TextUtils.isEmpty(text)){ |
||||||
|
tv.setText(text); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
//-------------------------------------------
|
||||||
|
|
||||||
|
public void dismissDialogFragment(FragmentManager fragmentManager){ |
||||||
|
dismissDialogFragment(fragmentManager,mTag); |
||||||
|
} |
||||||
|
|
||||||
|
public void dismissDialogFragment(FragmentManager fragmentManager,String tag){ |
||||||
|
if(tag!=null){ |
||||||
|
DialogFragment dialogFragment = (DialogFragment) fragmentManager.findFragmentByTag(tag); |
||||||
|
dismissDialogFragment(dialogFragment); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
public void dismissDialogFragment(DialogFragment dialogFragment){ |
||||||
|
if(dialogFragment!=null){ |
||||||
|
dialogFragment.dismiss(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
//-------------------------------------------
|
||||||
|
|
||||||
|
public String showDialogFragment(FragmentManager fragmentManager,AppDialogConfig config){ |
||||||
|
AppDialogFragment dialogFragment = AppDialogFragment.newInstance(config); |
||||||
|
String tag = dialogFragment.getTag() !=null ? dialogFragment.getTag() : dialogFragment.getClass().getSimpleName(); |
||||||
|
showDialogFragment(fragmentManager,dialogFragment,tag); |
||||||
|
mTag = tag; |
||||||
|
return tag; |
||||||
|
} |
||||||
|
|
||||||
|
public String showDialogFragment(FragmentManager fragmentManager,DialogFragment dialogFragment){ |
||||||
|
String tag = dialogFragment.getTag() !=null ? dialogFragment.getTag() : dialogFragment.getClass().getSimpleName(); |
||||||
|
showDialogFragment(fragmentManager,dialogFragment,tag); |
||||||
|
mTag = tag; |
||||||
|
return tag; |
||||||
|
} |
||||||
|
|
||||||
|
public String showDialogFragment(FragmentManager fragmentManager,DialogFragment dialogFragment, String tag) { |
||||||
|
dialogFragment.show(fragmentManager,tag); |
||||||
|
mTag = tag; |
||||||
|
return tag; |
||||||
|
} |
||||||
|
|
||||||
|
//-------------------------------------------
|
||||||
|
|
||||||
|
public void showDialog(Context context,AppDialogConfig config){ |
||||||
|
showDialog(context,createAppDialogView(context,config)); |
||||||
|
} |
||||||
|
|
||||||
|
public void showDialog(Context context,View contentView){ |
||||||
|
showDialog(context,contentView,DEFAULT_WIDTH_RATIO); |
||||||
|
} |
||||||
|
|
||||||
|
public void showDialog(Context context,View contentView,boolean isCancel){ |
||||||
|
showDialog(context,contentView,R.style.app_dialog,DEFAULT_WIDTH_RATIO,isCancel); |
||||||
|
} |
||||||
|
|
||||||
|
public void showDialog(Context context,View contentView,float widthRatio){ |
||||||
|
showDialog(context,contentView,widthRatio,true); |
||||||
|
} |
||||||
|
|
||||||
|
public void showDialog(Context context,View contentView,float widthRatio,boolean isCancel){ |
||||||
|
showDialog(context,contentView,R.style.app_dialog,widthRatio,isCancel); |
||||||
|
} |
||||||
|
|
||||||
|
public void showDialog(Context context, View contentView, @StyleRes int resId, float widthRatio){ |
||||||
|
showDialog(context,contentView,resId,widthRatio,true); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param context |
||||||
|
* @param contentView |
||||||
|
* @param resId Dialog样式 |
||||||
|
* @param widthRatio |
||||||
|
* @param isCancel 是否可取消(默认为true,false则拦截back键) |
||||||
|
*/ |
||||||
|
public void showDialog(Context context, View contentView, @StyleRes int resId, float widthRatio,final boolean isCancel){ |
||||||
|
mDialog = new Dialog(context,resId); |
||||||
|
mDialog.setContentView(contentView); |
||||||
|
mDialog.setCanceledOnTouchOutside(false); |
||||||
|
mDialog.setOnKeyListener(new DialogInterface.OnKeyListener() { |
||||||
|
@Override |
||||||
|
public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) { |
||||||
|
if(keyCode == KeyEvent.KEYCODE_BACK && isCancel){ |
||||||
|
dismissDialog(); |
||||||
|
} |
||||||
|
return true; |
||||||
|
|
||||||
|
} |
||||||
|
}); |
||||||
|
setDialogWindow(context,mDialog,widthRatio); |
||||||
|
mDialog.show(); |
||||||
|
} |
||||||
|
|
||||||
|
private void setDialogWindow(Context context,Dialog dialog,float widthRatio){ |
||||||
|
Window window = dialog.getWindow(); |
||||||
|
WindowManager.LayoutParams lp = window.getAttributes(); |
||||||
|
lp.width = (int)(context.getResources().getDisplayMetrics().widthPixels * widthRatio); |
||||||
|
window.setAttributes(lp); |
||||||
|
} |
||||||
|
|
||||||
|
public void dismissDialog(){ |
||||||
|
dismissDialog(mDialog); |
||||||
|
} |
||||||
|
|
||||||
|
private void dismissDialog(Dialog dialog){ |
||||||
|
if(dialog!=null){ |
||||||
|
dialog.dismiss(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
//-------------------------------------------
|
||||||
|
|
||||||
|
} |
@ -0,0 +1,156 @@ |
|||||||
|
package com.king.app.dialog; |
||||||
|
|
||||||
|
import android.support.annotation.IdRes; |
||||||
|
import android.support.annotation.LayoutRes; |
||||||
|
import android.view.View; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class AppDialogConfig { |
||||||
|
|
||||||
|
private @LayoutRes int layoutId = R.layout.app_dialog; |
||||||
|
|
||||||
|
private @IdRes int titleId = R.id.tvDialogTitle; |
||||||
|
|
||||||
|
private @IdRes int contentId = R.id.tvDialogContent; |
||||||
|
|
||||||
|
private @IdRes int cancelId = R.id.btnDialogCancel; |
||||||
|
|
||||||
|
private @IdRes int okId = R.id.btnDialogOK; |
||||||
|
|
||||||
|
private @IdRes int line = R.id.line; |
||||||
|
|
||||||
|
private CharSequence title; |
||||||
|
|
||||||
|
private CharSequence content; |
||||||
|
|
||||||
|
private CharSequence cancel; |
||||||
|
|
||||||
|
private CharSequence ok; |
||||||
|
|
||||||
|
private boolean isHideCancel; |
||||||
|
|
||||||
|
private View.OnClickListener onClickCancel; |
||||||
|
|
||||||
|
private View.OnClickListener onClickOk; |
||||||
|
|
||||||
|
public int getLayoutId() { |
||||||
|
return layoutId; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setLayoutId(int layoutId) { |
||||||
|
this.layoutId = layoutId; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public int getTitleId() { |
||||||
|
return titleId; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setTitleId(int titleId) { |
||||||
|
this.titleId = titleId; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public int getContentId() { |
||||||
|
return contentId; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setContentId(int contentId) { |
||||||
|
this.contentId = contentId; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public int getCancelId() { |
||||||
|
return cancelId; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setCancelId(int cancelId) { |
||||||
|
this.cancelId = cancelId; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public int getOkId() { |
||||||
|
return okId; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setOkId(int okId) { |
||||||
|
this.okId = okId; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public int getLine() { |
||||||
|
return line; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setLine(int line) { |
||||||
|
this.line = line; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public CharSequence getTitle() { |
||||||
|
return title; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setTitle(CharSequence title) { |
||||||
|
this.title = title; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public CharSequence getContent() { |
||||||
|
return content; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setContent(CharSequence content) { |
||||||
|
this.content = content; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public CharSequence getCancel() { |
||||||
|
return cancel; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setCancel(CharSequence cancel) { |
||||||
|
this.cancel = cancel; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public CharSequence getOk() { |
||||||
|
return ok; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setOk(CharSequence ok) { |
||||||
|
this.ok = ok; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public boolean isHideCancel() { |
||||||
|
return isHideCancel; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setHideCancel(boolean hideCancel) { |
||||||
|
isHideCancel = hideCancel; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public View.OnClickListener getOnClickCancel() { |
||||||
|
return onClickCancel; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setOnClickCancel(View.OnClickListener onClickCancel) { |
||||||
|
this.onClickCancel = onClickCancel; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public View.OnClickListener getOnClickOk() { |
||||||
|
return onClickOk; |
||||||
|
} |
||||||
|
|
||||||
|
public AppDialogConfig setOnClickOk(View.OnClickListener onClickOk) { |
||||||
|
this.onClickOk = onClickOk; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
} |
@ -0,0 +1,61 @@ |
|||||||
|
package com.king.app.dialog.fragment; |
||||||
|
|
||||||
|
import android.os.Bundle; |
||||||
|
import android.view.View; |
||||||
|
import android.widget.Button; |
||||||
|
import android.widget.TextView; |
||||||
|
|
||||||
|
import com.king.app.dialog.AppDialogConfig; |
||||||
|
import com.king.app.dialog.R; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class AppDialogFragment extends BaseDialogFragment { |
||||||
|
|
||||||
|
private AppDialogConfig config; |
||||||
|
|
||||||
|
public static AppDialogFragment newInstance(AppDialogConfig config) { |
||||||
|
|
||||||
|
Bundle args = new Bundle(); |
||||||
|
AppDialogFragment fragment = new AppDialogFragment(); |
||||||
|
fragment.config = config; |
||||||
|
fragment.setArguments(args); |
||||||
|
return fragment; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int getRootLayoutId() { |
||||||
|
if(config == null){ |
||||||
|
config = new AppDialogConfig(); |
||||||
|
} |
||||||
|
return config.getLayoutId(); |
||||||
|
} |
||||||
|
|
||||||
|
public void init(View rootView){ |
||||||
|
if(config!=null){ |
||||||
|
TextView tvDialogTitle = rootView.findViewById(config.getTitleId()); |
||||||
|
setText(tvDialogTitle,config.getTitle()); |
||||||
|
|
||||||
|
TextView tvDialogContent = rootView.findViewById(config.getContentId()); |
||||||
|
setText(tvDialogContent,config.getContent()); |
||||||
|
|
||||||
|
Button btnDialogCancel = rootView.findViewById(config.getCancelId()); |
||||||
|
setText(btnDialogCancel,config.getCancel()); |
||||||
|
btnDialogCancel.setOnClickListener(config.getOnClickCancel() != null ? config.getOnClickCancel() : getOnClickDismiss()); |
||||||
|
btnDialogCancel.setVisibility(config.isHideCancel() ? View.GONE : View.VISIBLE); |
||||||
|
|
||||||
|
try{ |
||||||
|
//不强制要求要有横线
|
||||||
|
View line = rootView.findViewById(R.id.line); |
||||||
|
line.setVisibility(config.isHideCancel() ? View.GONE : View.VISIBLE); |
||||||
|
}catch (Exception e){ |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
Button btnDialogOK = rootView.findViewById(config.getOkId()); |
||||||
|
setText(btnDialogOK,config.getOk()); |
||||||
|
btnDialogOK.setOnClickListener(config.getOnClickOk() != null ? config.getOnClickOk() : getOnClickDismiss()); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,59 @@ |
|||||||
|
package com.king.app.dialog.fragment; |
||||||
|
|
||||||
|
import android.graphics.Color; |
||||||
|
import android.graphics.drawable.ColorDrawable; |
||||||
|
import android.os.Bundle; |
||||||
|
import android.support.annotation.NonNull; |
||||||
|
import android.support.annotation.Nullable; |
||||||
|
import android.support.v4.app.DialogFragment; |
||||||
|
import android.text.TextUtils; |
||||||
|
import android.view.LayoutInflater; |
||||||
|
import android.view.View; |
||||||
|
import android.view.ViewGroup; |
||||||
|
import android.view.Window; |
||||||
|
import android.widget.TextView; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public abstract class BaseDialogFragment extends DialogFragment { |
||||||
|
|
||||||
|
private View mRootView; |
||||||
|
|
||||||
|
@Nullable |
||||||
|
@Override |
||||||
|
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { |
||||||
|
super.getDialog().requestWindowFeature(Window.FEATURE_NO_TITLE); |
||||||
|
super.getDialog().getWindow().setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT)); |
||||||
|
|
||||||
|
mRootView = inflater.inflate(getRootLayoutId(), container, false); |
||||||
|
init(mRootView); |
||||||
|
return mRootView; |
||||||
|
} |
||||||
|
|
||||||
|
protected View getRootView(){ |
||||||
|
return mRootView; |
||||||
|
} |
||||||
|
|
||||||
|
protected void setText(TextView tv, CharSequence text){ |
||||||
|
if(!TextUtils.isEmpty(text)){ |
||||||
|
tv.setText(text); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
protected View.OnClickListener getOnClickDismiss(){ |
||||||
|
return mOnClickDismissDialog; |
||||||
|
} |
||||||
|
|
||||||
|
private View.OnClickListener mOnClickDismissDialog = new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
dismiss(); |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
public abstract int getRootLayoutId(); |
||||||
|
|
||||||
|
public abstract void init(View rootView); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,10 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<set xmlns:android="http://schemas.android.com/apk/res/android"> |
||||||
|
<scale android:duration="200" |
||||||
|
android:pivotX="50%" |
||||||
|
android:pivotY="50%" |
||||||
|
android:fromXScale="0.0" |
||||||
|
android:fromYScale="0.0" |
||||||
|
android:toXScale="1.0" |
||||||
|
android:toYScale="1.0"/> |
||||||
|
</set> |
@ -0,0 +1,10 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<set xmlns:android="http://schemas.android.com/apk/res/android"> |
||||||
|
<scale android:duration="200" |
||||||
|
android:pivotX="50%" |
||||||
|
android:pivotY="50%" |
||||||
|
android:fromXScale="1.0" |
||||||
|
android:fromYScale="1.0" |
||||||
|
android:toXScale="0.0" |
||||||
|
android:toYScale="0.0"/> |
||||||
|
</set> |
@ -0,0 +1,5 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<selector xmlns:android="http://schemas.android.com/apk/res/android"> |
||||||
|
<item android:state_pressed="true" android:color="@color/app_dialog_button_pressed_color"/> |
||||||
|
<item android:state_pressed="false" android:color="@color/app_dialog_button_normal_color"/> |
||||||
|
</selector> |
@ -0,0 +1,12 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<shape xmlns:android="http://schemas.android.com/apk/res/android" > |
||||||
|
|
||||||
|
<solid android:color="@android:color/white" /> |
||||||
|
|
||||||
|
<corners |
||||||
|
android:bottomLeftRadius="10dp" |
||||||
|
android:bottomRightRadius="10dp" |
||||||
|
android:topLeftRadius="10dp" |
||||||
|
android:topRightRadius="10dp" /> |
||||||
|
|
||||||
|
</shape> |
@ -0,0 +1,60 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_gravity="center" |
||||||
|
android:layout_margin="30dp" |
||||||
|
android:background="@drawable/app_dialog_bg" |
||||||
|
android:orientation="vertical"> |
||||||
|
<TextView |
||||||
|
android:id="@+id/tvDialogTitle" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:gravity="center" |
||||||
|
android:minHeight="40dp" |
||||||
|
android:padding="6dp" |
||||||
|
android:lines="1" |
||||||
|
android:textSize="16sp" |
||||||
|
android:textColor="@color/app_dialog_title_color" |
||||||
|
android:text="@string/app_dialog_title"/> |
||||||
|
<TextView |
||||||
|
android:id="@+id/tvDialogContent" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:padding="10dp" |
||||||
|
android:layout_marginBottom="6dp" |
||||||
|
android:lineSpacingMultiplier="1" /> |
||||||
|
<include layout="@layout/app_dialog_line_h"/> |
||||||
|
<LinearLayout |
||||||
|
android:orientation="horizontal" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:gravity="center_horizontal"> |
||||||
|
<Button |
||||||
|
android:id="@+id/btnDialogCancel" |
||||||
|
android:layout_width="0dp" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:paddingTop="10dp" |
||||||
|
android:paddingBottom="10dp" |
||||||
|
android:layout_weight="1" |
||||||
|
android:text="@string/app_dialog_cancel" |
||||||
|
android:textSize="18sp" |
||||||
|
android:textColor="@color/app_dialog_button_color_selector" |
||||||
|
android:background="?android:attr/selectableItemBackground"/> |
||||||
|
<include |
||||||
|
android:id="@+id/line" |
||||||
|
layout="@layout/app_dialog_line_v"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btnDialogOK" |
||||||
|
android:layout_width="0dp" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:paddingTop="10dp" |
||||||
|
android:paddingBottom="10dp" |
||||||
|
android:layout_weight="1" |
||||||
|
android:text="@string/app_dialog_ok" |
||||||
|
android:textSize="18sp" |
||||||
|
android:textColor="@color/app_dialog_button_color_selector" |
||||||
|
android:background="?android:attr/selectableItemBackground"/> |
||||||
|
</LinearLayout> |
||||||
|
|
||||||
|
</LinearLayout> |
@ -0,0 +1,5 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<View xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="@dimen/app_dialog_line" |
||||||
|
android:background="@color/app_dialog_line_color"/> |
@ -0,0 +1,5 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<View xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
android:layout_width="@dimen/app_dialog_line" |
||||||
|
android:layout_height="match_parent" |
||||||
|
android:background="@color/app_dialog_line_color"/> |
@ -0,0 +1,10 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<resources> |
||||||
|
|
||||||
|
<color name="app_dialog_title_color">#333333</color> |
||||||
|
<color name="app_dialog_content_color">#666666</color> |
||||||
|
<color name="app_dialog_button_normal_color">#333333</color> |
||||||
|
<color name="app_dialog_button_pressed_color">#FF4081</color> |
||||||
|
|
||||||
|
<color name="app_dialog_line_color">#d2d2d2</color> |
||||||
|
</resources> |
@ -0,0 +1,5 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<resources> |
||||||
|
|
||||||
|
<dimen name="app_dialog_line">0.4dp</dimen> |
||||||
|
</resources> |
@ -0,0 +1,7 @@ |
|||||||
|
<resources> |
||||||
|
<string name="app_name">AppDialog</string> |
||||||
|
|
||||||
|
<string name="app_dialog_title">提示</string> |
||||||
|
<string name="app_dialog_cancel">取消</string> |
||||||
|
<string name="app_dialog_ok">确认</string> |
||||||
|
</resources> |
@ -0,0 +1,21 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<resources> |
||||||
|
|
||||||
|
<!-- dialog animation --> |
||||||
|
<style name="app_dialog_animation" parent="android:style/Animation.Dialog"> |
||||||
|
<item name="android:windowEnterAnimation">@anim/app_dialog_in</item> |
||||||
|
<item name="android:windowExitAnimation">@anim/app_dialog_out</item> |
||||||
|
</style> |
||||||
|
|
||||||
|
<!-- dialog style --> |
||||||
|
<style name="app_dialog" parent="android:style/Theme.Dialog"> |
||||||
|
<item name="android:windowFrame">@null</item> |
||||||
|
<item name="android:windowIsFloating">true</item> |
||||||
|
<item name="android:windowIsTranslucent">true</item> |
||||||
|
<item name="android:windowNoTitle">true</item> |
||||||
|
<item name="android:windowBackground">@android:color/transparent</item> |
||||||
|
<item name="android:backgroundDimEnabled">true</item> |
||||||
|
<item name="android:windowAnimationStyle">@style/app_dialog_animation</item> |
||||||
|
</style> |
||||||
|
|
||||||
|
</resources> |
@ -0,0 +1,17 @@ |
|||||||
|
package com.king.app.dialog; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
|
||||||
|
import static org.junit.Assert.*; |
||||||
|
|
||||||
|
/** |
||||||
|
* Example local unit test, which will execute on the development machine (host). |
||||||
|
* |
||||||
|
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a> |
||||||
|
*/ |
||||||
|
public class ExampleUnitTest { |
||||||
|
@Test |
||||||
|
public void addition_isCorrect() { |
||||||
|
assertEquals(4, 2 + 2); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1 @@ |
|||||||
|
/build |
@ -0,0 +1,11 @@ |
|||||||
|
apply plugin: 'com.novoda.bintray-release' |
||||||
|
|
||||||
|
//添加 |
||||||
|
publish { |
||||||
|
userOrg = 'jenly'//bintray.com用户名 |
||||||
|
groupId = 'com.king.app'//jcenter上的路径 |
||||||
|
artifactId = 'app-updater'//项目名称 |
||||||
|
publishVersion = '1.0'//版本号 |
||||||
|
desc = 'AppUpdater for Android'//描述 |
||||||
|
website = 'https://github.com/jenly1314/AppUpdater'//网站 |
||||||
|
} |
@ -0,0 +1,35 @@ |
|||||||
|
apply plugin: 'com.android.library' |
||||||
|
apply from: 'bintray.gradle' |
||||||
|
|
||||||
|
android { |
||||||
|
compileSdkVersion 27 |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
defaultConfig { |
||||||
|
minSdkVersion 15 |
||||||
|
targetSdkVersion 27 |
||||||
|
versionCode 1 |
||||||
|
versionName "1.0" |
||||||
|
|
||||||
|
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
buildTypes { |
||||||
|
release { |
||||||
|
minifyEnabled false |
||||||
|
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
dependencies { |
||||||
|
implementation fileTree(dir: 'libs', include: ['*.jar']) |
||||||
|
|
||||||
|
compileOnly 'com.android.support:appcompat-v7:27.1.1' |
||||||
|
testImplementation 'junit:junit:4.12' |
||||||
|
androidTestImplementation 'com.android.support.test:runner:1.0.2' |
||||||
|
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2' |
||||||
|
} |
@ -0,0 +1,21 @@ |
|||||||
|
# Add project specific ProGuard rules here. |
||||||
|
# You can control the set of applied configuration files using the |
||||||
|
# proguardFiles setting in build.gradle. |
||||||
|
# |
||||||
|
# For more details, see |
||||||
|
# http://developer.android.com/guide/developing/tools/proguard.html |
||||||
|
|
||||||
|
# If your project uses WebView with JS, uncomment the following |
||||||
|
# and specify the fully qualified class name to the JavaScript interface |
||||||
|
# class: |
||||||
|
#-keepclassmembers class fqcn.of.javascript.interface.for.webview { |
||||||
|
# public *; |
||||||
|
#} |
||||||
|
|
||||||
|
# Uncomment this to preserve the line number information for |
||||||
|
# debugging stack traces. |
||||||
|
#-keepattributes SourceFile,LineNumberTable |
||||||
|
|
||||||
|
# If you keep the line number information, uncomment this to |
||||||
|
# hide the original source file name. |
||||||
|
#-renamesourcefileattribute SourceFile |
@ -0,0 +1,26 @@ |
|||||||
|
package com.king.app.updater; |
||||||
|
|
||||||
|
import android.content.Context; |
||||||
|
import android.support.test.InstrumentationRegistry; |
||||||
|
import android.support.test.runner.AndroidJUnit4; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
import org.junit.runner.RunWith; |
||||||
|
|
||||||
|
import static org.junit.Assert.*; |
||||||
|
|
||||||
|
/** |
||||||
|
* Instrumented test, which will execute on an Android device. |
||||||
|
* |
||||||
|
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a> |
||||||
|
*/ |
||||||
|
@RunWith(AndroidJUnit4.class) |
||||||
|
public class ExampleInstrumentedTest { |
||||||
|
@Test |
||||||
|
public void useAppContext() { |
||||||
|
// Context of the app under test.
|
||||||
|
Context appContext = InstrumentationRegistry.getTargetContext(); |
||||||
|
|
||||||
|
assertEquals("com.king.app.updater.test", appContext.getPackageName()); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,27 @@ |
|||||||
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
package="com.king.app.updater"> |
||||||
|
|
||||||
|
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> |
||||||
|
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> |
||||||
|
|
||||||
|
<uses-permission android:name="android.permission.READ_PHONE_STATE" /> |
||||||
|
|
||||||
|
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" /> |
||||||
|
|
||||||
|
<application> |
||||||
|
|
||||||
|
<service android:name=".service.DownloadService"/> |
||||||
|
|
||||||
|
<provider |
||||||
|
android:name=".provider.AppUpdaterFileProvider" |
||||||
|
android:authorities="${applicationId}.fileProvider" |
||||||
|
android:exported="false" |
||||||
|
android:grantUriPermissions="true"> |
||||||
|
<meta-data |
||||||
|
android:name="android.support.FILE_PROVIDER_PATHS" |
||||||
|
android:resource="@xml/app_updater_paths"/> |
||||||
|
</provider> |
||||||
|
|
||||||
|
</application> |
||||||
|
|
||||||
|
</manifest> |
@ -0,0 +1,239 @@ |
|||||||
|
package com.king.app.updater; |
||||||
|
|
||||||
|
import android.app.Activity; |
||||||
|
import android.content.ComponentName; |
||||||
|
import android.content.Context; |
||||||
|
import android.content.Intent; |
||||||
|
import android.content.ServiceConnection; |
||||||
|
import android.os.IBinder; |
||||||
|
import android.support.annotation.DrawableRes; |
||||||
|
import android.support.annotation.NonNull; |
||||||
|
import android.text.TextUtils; |
||||||
|
|
||||||
|
import com.king.app.updater.callback.UpdateCallback; |
||||||
|
import com.king.app.updater.constant.Constants; |
||||||
|
import com.king.app.updater.http.IHttpManager; |
||||||
|
import com.king.app.updater.service.DownloadService; |
||||||
|
import com.king.app.updater.util.PermissionUtils; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class AppUpdater { |
||||||
|
|
||||||
|
private Context mContext; |
||||||
|
|
||||||
|
private UpdateConfig mConfig; |
||||||
|
|
||||||
|
private UpdateCallback mCallback; |
||||||
|
|
||||||
|
private IHttpManager mHttpManager; |
||||||
|
|
||||||
|
private ServiceConnection mServiceConnection; |
||||||
|
|
||||||
|
|
||||||
|
public AppUpdater(@NonNull Context context,@NonNull UpdateConfig config){ |
||||||
|
this.mContext = context; |
||||||
|
this.mConfig = config; |
||||||
|
} |
||||||
|
|
||||||
|
public AppUpdater(@NonNull Context context,@NonNull String url){ |
||||||
|
this.mContext = context; |
||||||
|
mConfig = new UpdateConfig(); |
||||||
|
mConfig.setUrl(url); |
||||||
|
} |
||||||
|
|
||||||
|
public AppUpdater setUpdateCallback(UpdateCallback callback){ |
||||||
|
this.mCallback = callback; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public AppUpdater setHttpManager(IHttpManager httpManager){ |
||||||
|
this.mHttpManager = httpManager; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 开始下载 |
||||||
|
*/ |
||||||
|
public void start(){ |
||||||
|
if(mConfig!=null && !TextUtils.isEmpty(mConfig.getUrl())){ |
||||||
|
//mContext不强制要求是Activity,能传Activity尽量传。AppUpdater本应该只专注于App更新,尽量不涉及动态权限相关的处理。如果mContext不强制要求是Activity是Activity默认会优先校验一次动态权限。
|
||||||
|
if(mContext instanceof Activity){ |
||||||
|
PermissionUtils.INSTANCE.verifyReadAndWritePermissions((Activity) mContext,Constants.RE_CODE_STORAGE_PERMISSION); |
||||||
|
} |
||||||
|
startDownloadService(); |
||||||
|
}else{ |
||||||
|
throw new NullPointerException("url = null"); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 启动下载服务 |
||||||
|
*/ |
||||||
|
private void startDownloadService(){ |
||||||
|
|
||||||
|
Intent intent = new Intent(mContext, DownloadService.class); |
||||||
|
if(mCallback!=null || mHttpManager!=null){//bindService
|
||||||
|
mServiceConnection = new ServiceConnection() { |
||||||
|
@Override |
||||||
|
public void onServiceConnected(ComponentName name, IBinder service) { |
||||||
|
DownloadService.DownloadBinder binder = ((DownloadService.DownloadBinder)service); |
||||||
|
binder.start(mConfig,mHttpManager,mCallback); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onServiceDisconnected(ComponentName name) { |
||||||
|
|
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
mContext.getApplicationContext().bindService(intent,mServiceConnection, Context.BIND_AUTO_CREATE); |
||||||
|
}else{//startService
|
||||||
|
intent.putExtra(Constants.KEY_UPDATE_CONFIG,mConfig); |
||||||
|
mContext.startService(intent); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* AppUpdater构建器 |
||||||
|
*/ |
||||||
|
public static class Builder{ |
||||||
|
|
||||||
|
private UpdateConfig mConfig; |
||||||
|
|
||||||
|
public Builder(){ |
||||||
|
mConfig = new UpdateConfig(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param url 下载地址 |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder serUrl(@NonNull String url){ |
||||||
|
mConfig.setUrl(url); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param path 下载保存的文件路径 (默认SD卡/.AppUpdater目录) |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setPath(String path){ |
||||||
|
mConfig.setPath(path); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param filename 下载的保存的apk文件名 (默认优先取url文件名) |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setFilename(String filename){ |
||||||
|
mConfig.setFilename(filename); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param isShowNotification 是否显示通知栏 (默认true) |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setShowNotification(boolean isShowNotification) { |
||||||
|
mConfig.setShowNotification(isShowNotification); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param notifyId 通知ID |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setNotificationId(int notifyId) { |
||||||
|
mConfig.setNotificationId(notifyId); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param channelId 通知渠道ID (默认兼容O) |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setChannelId(String channelId) { |
||||||
|
mConfig.setChannelId(channelId); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param channelName 通知渠道名称 (默认兼容O) |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setChannelName(String channelName) { |
||||||
|
mConfig.setChannelName(channelName); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param icon 通知栏图标 (默认取App的icon) |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setNotificationIcon(@DrawableRes int icon) { |
||||||
|
mConfig.setNotificationIcon(icon); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param isInstallApk 下载完成后是否自动调用安装APK(默认true) |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setInstallApk(boolean isInstallApk){ |
||||||
|
mConfig.setInstallApk(isInstallApk); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param authority FileProvider的authority(默认兼容N,默认值context.getPackageName() + ".fileProvider") |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setAuthority(String authority){ |
||||||
|
mConfig.setAuthority(authority); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param showPercentage 下载时通知栏是否显示百分比 |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setShowPercentage(boolean showPercentage) { |
||||||
|
mConfig.setShowPercentage(showPercentage); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param reDownload 下载失败时是否支持点击通知栏重新下载,默认true 最多重新下载3次 |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public Builder setReDownload(boolean reDownload) { |
||||||
|
mConfig.setReDownload(reDownload); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public AppUpdater build(@NonNull Context context){ |
||||||
|
AppUpdater appUpdater = new AppUpdater(context,mConfig); |
||||||
|
return appUpdater; |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
} |
@ -0,0 +1,218 @@ |
|||||||
|
package com.king.app.updater; |
||||||
|
|
||||||
|
|
||||||
|
import android.os.Parcel; |
||||||
|
import android.os.Parcelable; |
||||||
|
import android.support.annotation.DrawableRes; |
||||||
|
|
||||||
|
import com.king.app.updater.constant.Constants; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class UpdateConfig implements Parcelable { |
||||||
|
|
||||||
|
|
||||||
|
private String mUrl; |
||||||
|
/** |
||||||
|
* 保存路径 |
||||||
|
*/ |
||||||
|
private String mPath; |
||||||
|
/** |
||||||
|
* 保存文件名 |
||||||
|
*/ |
||||||
|
private String mFilename; |
||||||
|
|
||||||
|
private boolean isShowNotification = true; |
||||||
|
/** |
||||||
|
* 下载完成后是否自动弹出安装 |
||||||
|
*/ |
||||||
|
private boolean isInstallApk = true; |
||||||
|
|
||||||
|
private int mNotificationIcon; |
||||||
|
|
||||||
|
private int mNotificationId = Constants.DEFAULT_NOTIFICATION_ID; |
||||||
|
|
||||||
|
private String mChannelId; |
||||||
|
|
||||||
|
private String mChannelName; |
||||||
|
|
||||||
|
private String mAuthority; |
||||||
|
/** |
||||||
|
* 下载失败是否支持点击通知栏重复下载 |
||||||
|
*/ |
||||||
|
private boolean isReDownload = true; |
||||||
|
/** |
||||||
|
* 是否显示百分比 |
||||||
|
*/ |
||||||
|
private boolean isShowPercentage = true; |
||||||
|
|
||||||
|
|
||||||
|
public UpdateConfig() { |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
public String getUrl() { |
||||||
|
return mUrl; |
||||||
|
} |
||||||
|
|
||||||
|
public void setUrl(String url) { |
||||||
|
this.mUrl = url; |
||||||
|
} |
||||||
|
|
||||||
|
public String getPath() { |
||||||
|
return mPath; |
||||||
|
} |
||||||
|
|
||||||
|
public void setPath(String path) { |
||||||
|
this.mPath = path; |
||||||
|
} |
||||||
|
|
||||||
|
public String getFilename() { |
||||||
|
return mFilename; |
||||||
|
} |
||||||
|
|
||||||
|
public void setFilename(String filename) { |
||||||
|
this.mFilename = filename; |
||||||
|
} |
||||||
|
|
||||||
|
public boolean isShowNotification() { |
||||||
|
return isShowNotification; |
||||||
|
} |
||||||
|
|
||||||
|
public void setShowNotification(boolean isShowNotification) { |
||||||
|
this.isShowNotification = isShowNotification; |
||||||
|
} |
||||||
|
|
||||||
|
public String getChannelId() { |
||||||
|
return mChannelId; |
||||||
|
} |
||||||
|
|
||||||
|
public void setChannelId(String channelId) { |
||||||
|
this.mChannelId = channelId; |
||||||
|
} |
||||||
|
|
||||||
|
public String getChannelName() { |
||||||
|
return mChannelName; |
||||||
|
} |
||||||
|
|
||||||
|
public void setChannelName(String channelName) { |
||||||
|
this.mChannelName = channelName; |
||||||
|
} |
||||||
|
|
||||||
|
public void setNotificationId(int notificationId){ |
||||||
|
this.mNotificationId = notificationId; |
||||||
|
} |
||||||
|
|
||||||
|
public int getNotificationId(){ |
||||||
|
return this.mNotificationId; |
||||||
|
} |
||||||
|
|
||||||
|
public void setNotificationIcon(@DrawableRes int icon){ |
||||||
|
this.mNotificationIcon = icon; |
||||||
|
} |
||||||
|
|
||||||
|
public int getNotificationIcon(){ |
||||||
|
return this.mNotificationIcon; |
||||||
|
} |
||||||
|
|
||||||
|
public boolean isInstallApk() { |
||||||
|
return isInstallApk; |
||||||
|
} |
||||||
|
|
||||||
|
public void setInstallApk(boolean isInstallApk) { |
||||||
|
this.isInstallApk = isInstallApk; |
||||||
|
} |
||||||
|
|
||||||
|
public String getAuthority() { |
||||||
|
return mAuthority; |
||||||
|
} |
||||||
|
|
||||||
|
public void setAuthority(String authority) { |
||||||
|
this.mAuthority = authority; |
||||||
|
} |
||||||
|
|
||||||
|
public boolean isShowPercentage() { |
||||||
|
return isShowPercentage; |
||||||
|
} |
||||||
|
|
||||||
|
public void setShowPercentage(boolean showPercentage) { |
||||||
|
isShowPercentage = showPercentage; |
||||||
|
} |
||||||
|
|
||||||
|
public boolean isReDownload() { |
||||||
|
return isReDownload; |
||||||
|
} |
||||||
|
|
||||||
|
public void setReDownload(boolean reDownload) { |
||||||
|
isReDownload = reDownload; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public String toString() { |
||||||
|
return "UpdateConfig{" + |
||||||
|
"mUrl='" + mUrl + '\'' + |
||||||
|
", mPath='" + mPath + '\'' + |
||||||
|
", mFilename='" + mFilename + '\'' + |
||||||
|
", isShowNotification=" + isShowNotification + |
||||||
|
", isInstallApk=" + isInstallApk + |
||||||
|
", mNotificationIcon=" + mNotificationIcon + |
||||||
|
", mNotificationId=" + mNotificationId + |
||||||
|
", mChannelId='" + mChannelId + '\'' + |
||||||
|
", mChannelName='" + mChannelName + '\'' + |
||||||
|
", mAuthority='" + mAuthority + '\'' + |
||||||
|
", isShowPercentage=" + isShowPercentage + |
||||||
|
", isReDownload=" + isReDownload + |
||||||
|
'}'; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int describeContents() { |
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void writeToParcel(Parcel dest, int flags) { |
||||||
|
dest.writeString(this.mUrl); |
||||||
|
dest.writeString(this.mPath); |
||||||
|
dest.writeString(this.mFilename); |
||||||
|
dest.writeByte(this.isShowNotification ? (byte) 1 : (byte) 0); |
||||||
|
dest.writeByte(this.isInstallApk ? (byte) 1 : (byte) 0); |
||||||
|
dest.writeInt(this.mNotificationIcon); |
||||||
|
dest.writeInt(this.mNotificationId); |
||||||
|
dest.writeString(this.mChannelId); |
||||||
|
dest.writeString(this.mChannelName); |
||||||
|
dest.writeString(this.mAuthority); |
||||||
|
dest.writeByte(this.isReDownload ? (byte) 1 : (byte) 0); |
||||||
|
dest.writeByte(this.isShowPercentage ? (byte) 1 : (byte) 0); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
protected UpdateConfig(Parcel in) { |
||||||
|
this.mUrl = in.readString(); |
||||||
|
this.mPath = in.readString(); |
||||||
|
this.mFilename = in.readString(); |
||||||
|
this.isShowNotification = in.readByte() != 0; |
||||||
|
this.isInstallApk = in.readByte() != 0; |
||||||
|
this.mNotificationIcon = in.readInt(); |
||||||
|
this.mNotificationId = in.readInt(); |
||||||
|
this.mChannelId = in.readString(); |
||||||
|
this.mChannelName = in.readString(); |
||||||
|
this.mAuthority = in.readString(); |
||||||
|
this.isReDownload = in.readByte() != 0; |
||||||
|
this.isShowPercentage = in.readByte() != 0; |
||||||
|
} |
||||||
|
|
||||||
|
public static final Creator<UpdateConfig> CREATOR = new Creator<UpdateConfig>() { |
||||||
|
@Override |
||||||
|
public UpdateConfig createFromParcel(Parcel source) { |
||||||
|
return new UpdateConfig(source); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public UpdateConfig[] newArray(int size) { |
||||||
|
return new UpdateConfig[size]; |
||||||
|
} |
||||||
|
}; |
||||||
|
} |
@ -0,0 +1,26 @@ |
|||||||
|
package com.king.app.updater.callback; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public abstract class AppUpdateCallback implements UpdateCallback { |
||||||
|
@Override |
||||||
|
public void onDownloading(boolean isDownloading) { |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onStart(String url) { |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onError(Exception e) { |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onCancel() { |
||||||
|
|
||||||
|
} |
||||||
|
} |
@ -0,0 +1,46 @@ |
|||||||
|
package com.king.app.updater.callback; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public interface UpdateCallback { |
||||||
|
|
||||||
|
/** |
||||||
|
* 最开始调用(在onStart之前调用) |
||||||
|
* @param isDownloading true 表示已经在下载,false表示准备刚调用下载 |
||||||
|
*/ |
||||||
|
void onDownloading(boolean isDownloading); |
||||||
|
|
||||||
|
/** |
||||||
|
* 开始 |
||||||
|
*/ |
||||||
|
void onStart(String url); |
||||||
|
|
||||||
|
/** |
||||||
|
* 加载进度… |
||||||
|
* @param progress |
||||||
|
* @param total |
||||||
|
* @param isChange 进度百分比是否有改变,(主要可以用来过滤无用的刷新,从而降低刷新频率) |
||||||
|
*/ |
||||||
|
void onProgress(int progress,int total,boolean isChange); |
||||||
|
|
||||||
|
/** |
||||||
|
* 完成 |
||||||
|
* @param file |
||||||
|
*/ |
||||||
|
void onFinish(File file); |
||||||
|
|
||||||
|
/** |
||||||
|
* 错误 |
||||||
|
* @param e |
||||||
|
*/ |
||||||
|
void onError(Exception e); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* 取消 |
||||||
|
*/ |
||||||
|
void onCancel(); |
||||||
|
} |
@ -0,0 +1,31 @@ |
|||||||
|
package com.king.app.updater.constant; |
||||||
|
|
||||||
|
import android.os.Environment; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public final class Constants { |
||||||
|
|
||||||
|
public static final String TAG = "AppUpdater"; |
||||||
|
|
||||||
|
public static final String KEY_UPDATE_CONFIG = "app_update_config"; |
||||||
|
|
||||||
|
public static final int DEFAULT_NOTIFICATION_ID = 0x66; |
||||||
|
|
||||||
|
public static final String DEFAULT_NOTIFICATION_CHANNEL_ID = "0x66"; |
||||||
|
|
||||||
|
public static final String DEFAULT_NOTIFICATION_CHANNEL_NAME = "AppUpdater"; |
||||||
|
|
||||||
|
public static final String KEY_STOP_DOWNLOAD_SERVICE = "stop_download_service"; |
||||||
|
|
||||||
|
public static final String KEY_RE_DOWNLOAD = "app_update_re_download"; |
||||||
|
|
||||||
|
public static final String DEFAULT_DIR_PATH = Environment.getExternalStorageDirectory() + File.separator + ".AppUpdater"; |
||||||
|
|
||||||
|
public static final int RE_CODE_STORAGE_PERMISSION = 0x66; |
||||||
|
|
||||||
|
public static final int NONE = -1; |
||||||
|
} |
@ -0,0 +1,166 @@ |
|||||||
|
package com.king.app.updater.http; |
||||||
|
|
||||||
|
import android.os.AsyncTask; |
||||||
|
|
||||||
|
import com.king.app.updater.util.SSLSocketFactoryUtils; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.io.FileOutputStream; |
||||||
|
import java.io.InputStream; |
||||||
|
import java.net.ConnectException; |
||||||
|
import java.net.HttpURLConnection; |
||||||
|
import java.net.URL; |
||||||
|
|
||||||
|
import javax.net.ssl.HttpsURLConnection; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class HttpManager implements IHttpManager { |
||||||
|
|
||||||
|
private static final int DEFAULT_TIME_OUT = 20000; |
||||||
|
|
||||||
|
private int mTimeout = DEFAULT_TIME_OUT; |
||||||
|
|
||||||
|
private static HttpManager INSTANCE; |
||||||
|
|
||||||
|
public static HttpManager getInstance(){ |
||||||
|
if(INSTANCE == null){ |
||||||
|
synchronized (HttpManager.class){ |
||||||
|
INSTANCE = new HttpManager(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return INSTANCE; |
||||||
|
} |
||||||
|
|
||||||
|
private HttpManager(){ |
||||||
|
this(DEFAULT_TIME_OUT); |
||||||
|
} |
||||||
|
|
||||||
|
public HttpManager(int timeout){ |
||||||
|
this.mTimeout = timeout; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void download(String url, String path, String filename, DownloadCallback callback) { |
||||||
|
new DownloadTask(url,path,filename,callback).execute(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* 异步下载任务 |
||||||
|
*/ |
||||||
|
private class DownloadTask extends AsyncTask<Void,Integer,File> { |
||||||
|
private String url; |
||||||
|
|
||||||
|
private String path; |
||||||
|
|
||||||
|
private String filename; |
||||||
|
|
||||||
|
private DownloadCallback callback; |
||||||
|
|
||||||
|
private Exception exception; |
||||||
|
|
||||||
|
public DownloadTask(String url,String path,String filename,DownloadCallback callback){ |
||||||
|
this.url = url; |
||||||
|
this.path = path; |
||||||
|
this.filename = filename; |
||||||
|
this.callback = callback; |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
protected File doInBackground(Void... voids) { |
||||||
|
|
||||||
|
try { |
||||||
|
HttpsURLConnection.setDefaultSSLSocketFactory(SSLSocketFactoryUtils.createSSLSocketFactory()); |
||||||
|
HttpsURLConnection.setDefaultHostnameVerifier(SSLSocketFactoryUtils.createTrustAllHostnameVerifier()); |
||||||
|
HttpURLConnection connect = (HttpURLConnection)new URL(url).openConnection(); |
||||||
|
connect.setRequestMethod("GET"); |
||||||
|
connect.setReadTimeout(mTimeout); |
||||||
|
connect.setConnectTimeout(mTimeout); |
||||||
|
connect.connect(); |
||||||
|
int responseCode = connect.getResponseCode(); |
||||||
|
|
||||||
|
if(responseCode == HttpURLConnection.HTTP_OK){ |
||||||
|
|
||||||
|
InputStream is = connect.getInputStream(); |
||||||
|
|
||||||
|
int length = connect.getContentLength(); |
||||||
|
|
||||||
|
int progress = 0; |
||||||
|
|
||||||
|
byte[] buffer = new byte[4096]; |
||||||
|
|
||||||
|
int len; |
||||||
|
File file = new File(path,filename); |
||||||
|
FileOutputStream fos = new FileOutputStream(file); |
||||||
|
while ((len = is.read(buffer)) != -1){ |
||||||
|
fos.write(buffer,0,len); |
||||||
|
progress += len; |
||||||
|
//更新进度
|
||||||
|
publishProgress(progress,length); |
||||||
|
} |
||||||
|
|
||||||
|
fos.flush(); |
||||||
|
fos.close(); |
||||||
|
is.close(); |
||||||
|
|
||||||
|
connect.disconnect(); |
||||||
|
|
||||||
|
return file; |
||||||
|
}else {//连接失败
|
||||||
|
throw new ConnectException(String.format("responseCode = %d",responseCode)); |
||||||
|
} |
||||||
|
|
||||||
|
} catch (Exception e) { |
||||||
|
this.exception = e; |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
|
||||||
|
return null; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
protected void onPreExecute() { |
||||||
|
super.onPreExecute(); |
||||||
|
if(callback!=null){ |
||||||
|
callback.onStart(url); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
protected void onPostExecute(File file) { |
||||||
|
super.onPostExecute(file); |
||||||
|
if(callback!=null){ |
||||||
|
if(file!=null){ |
||||||
|
callback.onFinish(file); |
||||||
|
}else{ |
||||||
|
callback.onError(exception); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
protected void onProgressUpdate(Integer... values) { |
||||||
|
super.onProgressUpdate(values); |
||||||
|
if(callback!=null){ |
||||||
|
callback.onProgress(values[0],values[1]); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void onCancelled() { |
||||||
|
super.onCancelled(); |
||||||
|
if(callback!=null){ |
||||||
|
callback.onCancel(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,54 @@ |
|||||||
|
package com.king.app.updater.http; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.io.Serializable; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public interface IHttpManager { |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* 下载 |
||||||
|
* @param url |
||||||
|
* @param path |
||||||
|
* @param filename |
||||||
|
* @param callback |
||||||
|
*/ |
||||||
|
void download(String url,String path,String filename,DownloadCallback callback); |
||||||
|
|
||||||
|
|
||||||
|
interface DownloadCallback extends Serializable{ |
||||||
|
/** |
||||||
|
* 开始 |
||||||
|
* @param url |
||||||
|
*/ |
||||||
|
void onStart(String url); |
||||||
|
|
||||||
|
/** |
||||||
|
* 加载进度… |
||||||
|
* @param progress |
||||||
|
* @param total |
||||||
|
*/ |
||||||
|
void onProgress(int progress,int total); |
||||||
|
|
||||||
|
/** |
||||||
|
* 完成 |
||||||
|
* @param file |
||||||
|
*/ |
||||||
|
void onFinish(File file); |
||||||
|
|
||||||
|
/** |
||||||
|
* 错误 |
||||||
|
* @param e |
||||||
|
*/ |
||||||
|
void onError(Exception e); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* 取消 |
||||||
|
*/ |
||||||
|
void onCancel(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,10 @@ |
|||||||
|
package com.king.app.updater.provider; |
||||||
|
|
||||||
|
import android.support.v4.content.FileProvider; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class AppUpdaterFileProvider extends FileProvider { |
||||||
|
|
||||||
|
} |
@ -0,0 +1,527 @@ |
|||||||
|
package com.king.app.updater.service; |
||||||
|
|
||||||
|
import android.app.Notification; |
||||||
|
import android.app.NotificationChannel; |
||||||
|
import android.app.NotificationManager; |
||||||
|
import android.app.PendingIntent; |
||||||
|
import android.app.Service; |
||||||
|
import android.content.Context; |
||||||
|
import android.content.Intent; |
||||||
|
import android.net.Uri; |
||||||
|
import android.os.Binder; |
||||||
|
import android.os.Build; |
||||||
|
import android.os.Environment; |
||||||
|
import android.os.IBinder; |
||||||
|
import android.support.annotation.DrawableRes; |
||||||
|
import android.support.annotation.Nullable; |
||||||
|
import android.support.annotation.RequiresApi; |
||||||
|
import android.support.v4.app.NotificationCompat; |
||||||
|
import android.support.v4.content.FileProvider; |
||||||
|
import android.text.TextUtils; |
||||||
|
import android.util.Log; |
||||||
|
|
||||||
|
import com.king.app.updater.R; |
||||||
|
import com.king.app.updater.UpdateConfig; |
||||||
|
import com.king.app.updater.callback.UpdateCallback; |
||||||
|
import com.king.app.updater.constant.Constants; |
||||||
|
import com.king.app.updater.http.HttpManager; |
||||||
|
import com.king.app.updater.http.IHttpManager; |
||||||
|
import com.king.app.updater.util.AppUtils; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class DownloadService extends Service { |
||||||
|
|
||||||
|
private DownloadBinder mDownloadBinder = new DownloadBinder(); |
||||||
|
/** |
||||||
|
* 是否在下载,防止重复下载。 |
||||||
|
*/ |
||||||
|
private boolean isDownloading; |
||||||
|
/** |
||||||
|
* 最后更新进度,用来降频刷新 |
||||||
|
*/ |
||||||
|
private int mLastProgress = -1; |
||||||
|
/** |
||||||
|
* 最后进度更新时间,用来降频刷新 |
||||||
|
*/ |
||||||
|
private long mLastTime; |
||||||
|
/** |
||||||
|
* 失败后重新下载次数 |
||||||
|
*/ |
||||||
|
private int mCount = 0; |
||||||
|
|
||||||
|
private Context getContext(){ |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int onStartCommand(Intent intent, int flags, int startId) { |
||||||
|
|
||||||
|
if(intent != null){ |
||||||
|
if(!isDownloading){ |
||||||
|
boolean isStop = intent.getBooleanExtra(Constants.KEY_STOP_DOWNLOAD_SERVICE,false); |
||||||
|
if(isStop){ |
||||||
|
stopService(); |
||||||
|
}else{ |
||||||
|
//是否实通过通知栏触发重复下载
|
||||||
|
boolean isReDownload = intent.getBooleanExtra(Constants.KEY_RE_DOWNLOAD,false); |
||||||
|
if(isReDownload){ |
||||||
|
mCount++; |
||||||
|
} |
||||||
|
//获取配置信息
|
||||||
|
UpdateConfig config = intent.getParcelableExtra(Constants.KEY_UPDATE_CONFIG); |
||||||
|
startDownload(config,null,null); |
||||||
|
} |
||||||
|
}else{ |
||||||
|
Log.w(Constants.TAG,"已经在下载中,请勿重复下载。"); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return super.onStartCommand(intent, flags, startId); |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
//----------------------------------------
|
||||||
|
|
||||||
|
/** |
||||||
|
* 开始下载 |
||||||
|
* @param config |
||||||
|
* @param httpManager |
||||||
|
* @param callback |
||||||
|
*/ |
||||||
|
public void startDownload(UpdateConfig config,IHttpManager httpManager,UpdateCallback callback){ |
||||||
|
|
||||||
|
if(config == null){ |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
if(callback!=null){ |
||||||
|
callback.onDownloading(isDownloading); |
||||||
|
} |
||||||
|
|
||||||
|
if(isDownloading){ |
||||||
|
Log.w(Constants.TAG,"已经在下载中,请勿重复下载。"); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
String url = config.getUrl(); |
||||||
|
String path = config.getPath(); |
||||||
|
String filename = config.getFilename(); |
||||||
|
|
||||||
|
//如果保存路径为空则使用缓存路径
|
||||||
|
if(TextUtils.isEmpty(path)){ |
||||||
|
path = getDiskCacheDir(getContext()); |
||||||
|
} |
||||||
|
File dirFile = new File(path); |
||||||
|
if(!dirFile.exists()){ |
||||||
|
dirFile.mkdir(); |
||||||
|
} |
||||||
|
|
||||||
|
//如果文件名为空则使用路径
|
||||||
|
if(TextUtils.isEmpty(filename)){ |
||||||
|
filename = AppUtils.INSTANCE.getAppFullName(getContext(),url,getResources().getString(R.string.app_name)); |
||||||
|
} |
||||||
|
|
||||||
|
File file = new File(path,filename); |
||||||
|
if(file.exists()){ |
||||||
|
file.delete(); |
||||||
|
} |
||||||
|
|
||||||
|
if(httpManager != null){ |
||||||
|
httpManager.download(url,path,filename,new AppDownloadCallback(config,callback)); |
||||||
|
}else{ |
||||||
|
HttpManager.getInstance().download(url,path,filename,new AppDownloadCallback(config,callback)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获取缓存路径 |
||||||
|
* @param context |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public String getDiskCacheDir(Context context) { |
||||||
|
if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())) { |
||||||
|
return Constants.DEFAULT_DIR_PATH; |
||||||
|
} |
||||||
|
|
||||||
|
return context.getCacheDir().getAbsolutePath(); |
||||||
|
} |
||||||
|
|
||||||
|
private void stopService(){ |
||||||
|
mCount = 0; |
||||||
|
stopSelf(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
//---------------------------------------- DownloadCallback
|
||||||
|
|
||||||
|
public class AppDownloadCallback implements IHttpManager.DownloadCallback{ |
||||||
|
|
||||||
|
public UpdateConfig config; |
||||||
|
|
||||||
|
private boolean isShowNotification; |
||||||
|
|
||||||
|
private int notifyId; |
||||||
|
|
||||||
|
private String channelId; |
||||||
|
|
||||||
|
private String channelName; |
||||||
|
|
||||||
|
private int notificationIcon; |
||||||
|
|
||||||
|
private boolean isInstallApk; |
||||||
|
|
||||||
|
private String authority; |
||||||
|
|
||||||
|
private boolean isShowPercentage; |
||||||
|
|
||||||
|
private boolean isReDownload; |
||||||
|
|
||||||
|
private UpdateCallback callback; |
||||||
|
|
||||||
|
|
||||||
|
private AppDownloadCallback(UpdateConfig config,UpdateCallback callback){ |
||||||
|
this.config = config; |
||||||
|
this.callback = callback; |
||||||
|
this.isShowNotification = config.isShowNotification(); |
||||||
|
this.notifyId = config.getNotificationId(); |
||||||
|
|
||||||
|
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){ |
||||||
|
this.channelId = TextUtils.isEmpty(config.getChannelId()) ? Constants.DEFAULT_NOTIFICATION_CHANNEL_ID : config.getChannelId(); |
||||||
|
this.channelName = TextUtils.isEmpty(config.getChannelName()) ? Constants.DEFAULT_NOTIFICATION_CHANNEL_NAME : config.getChannelName(); |
||||||
|
} |
||||||
|
if(config.getNotificationIcon()<=0){ |
||||||
|
this.notificationIcon = AppUtils.INSTANCE.getAppIcon(getContext()); |
||||||
|
}else{ |
||||||
|
this.notificationIcon = config.getNotificationIcon(); |
||||||
|
} |
||||||
|
|
||||||
|
this.isInstallApk = config.isInstallApk(); |
||||||
|
if(TextUtils.isEmpty(config.getAuthority())){//如果为空默认
|
||||||
|
authority = getContext().getPackageName() + ".fileProvider"; |
||||||
|
}else{ |
||||||
|
this.authority = config.getAuthority(); |
||||||
|
} |
||||||
|
|
||||||
|
this.isShowPercentage = config.isShowPercentage(); |
||||||
|
this.isReDownload = config.isReDownload(); |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void onStart(String url) { |
||||||
|
isDownloading = true; |
||||||
|
mLastProgress = 0; |
||||||
|
if(isShowNotification){ |
||||||
|
showStartNotification(notifyId,channelId,channelName,notificationIcon,getString(R.string.app_updater_start_notification_title),getString(R.string.app_updater_start_notification_content)); |
||||||
|
} |
||||||
|
|
||||||
|
if(callback!=null){ |
||||||
|
callback.onStart(url); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onProgress(int progress, int total) { |
||||||
|
|
||||||
|
boolean isChange = false; |
||||||
|
long curTime = System.currentTimeMillis(); |
||||||
|
if(mLastTime + 200 < curTime) {//降低更新频率
|
||||||
|
mLastTime = curTime; |
||||||
|
|
||||||
|
int currProgress = Math.round(progress * 1.0f / total * 100.0f); |
||||||
|
if(currProgress!=mLastProgress){//百分比改变了才更新
|
||||||
|
isChange = true; |
||||||
|
String percentage = currProgress + "%"; |
||||||
|
if(isShowNotification) { |
||||||
|
mLastProgress = currProgress; |
||||||
|
String content = getString(R.string.app_updater_progress_notification_content); |
||||||
|
if (isShowPercentage) { |
||||||
|
content += percentage; |
||||||
|
} |
||||||
|
|
||||||
|
showProgressNotification(notifyId, channelId, notificationIcon, getString(R.string.app_updater_progress_notification_title), content, progress, total); |
||||||
|
|
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
if(callback!=null){ |
||||||
|
callback.onProgress(progress,total,isChange); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onFinish(File file) { |
||||||
|
isDownloading = false; |
||||||
|
showFinishNotification(notifyId,channelId,notificationIcon,getString(R.string.app_updater_finish_notification_title),getString(R.string.app_updater_finish_notification_content),file,authority); |
||||||
|
if(isInstallApk){ |
||||||
|
AppUtils.INSTANCE.installApk(getContext(),file,authority); |
||||||
|
} |
||||||
|
if(callback!=null){ |
||||||
|
callback.onFinish(file); |
||||||
|
} |
||||||
|
stopService(); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onError(Exception e) { |
||||||
|
isDownloading = false; |
||||||
|
//支持下载失败重新并最多支持失败下载3次
|
||||||
|
boolean isReDownload = this.isReDownload && mCount < 3; |
||||||
|
String content = isReDownload ? getString(R.string.app_updater_error_notification_content_re_download) : getString(R.string.app_updater_error_notification_content); |
||||||
|
showErrorNotification(notifyId,channelId,notificationIcon,getString(R.string.app_updater_error_notification_title),content,isReDownload,config); |
||||||
|
|
||||||
|
if(callback!=null){ |
||||||
|
callback.onError(e); |
||||||
|
} |
||||||
|
if(!isReDownload){ |
||||||
|
stopService(); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onCancel() { |
||||||
|
isDownloading = false; |
||||||
|
cancelNotification(notifyId); |
||||||
|
if(callback!=null){ |
||||||
|
callback.onCancel(); |
||||||
|
} |
||||||
|
stopService(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean onUnbind(Intent intent) { |
||||||
|
Log.d(Constants.TAG,"onUnbind"); |
||||||
|
return super.onUnbind(intent); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onDestroy() { |
||||||
|
isDownloading = false; |
||||||
|
Log.d(Constants.TAG,"onDestroy"); |
||||||
|
super.onDestroy(); |
||||||
|
} |
||||||
|
|
||||||
|
//---------------------------------------- Notification
|
||||||
|
|
||||||
|
/** |
||||||
|
* 显示开始下载是的通知 |
||||||
|
* @param notifyId |
||||||
|
* @param channelId |
||||||
|
* @param channelName |
||||||
|
* @param icon |
||||||
|
* @param title |
||||||
|
* @param content |
||||||
|
*/ |
||||||
|
private void showStartNotification(int notifyId,String channelId, String channelName,@DrawableRes int icon,CharSequence title,CharSequence content){ |
||||||
|
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){ |
||||||
|
createNotificationChannel(channelId,channelName); |
||||||
|
} |
||||||
|
NotificationCompat.Builder builder = buildNotification(channelId,icon,title,content); |
||||||
|
builder.setAutoCancel(false); |
||||||
|
notifyNotification(notifyId,builder.build()); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 显示下载中的通知(更新进度) |
||||||
|
* @param notifyId |
||||||
|
* @param channelId |
||||||
|
* @param icon |
||||||
|
* @param title |
||||||
|
* @param content |
||||||
|
* @param progress |
||||||
|
* @param size |
||||||
|
*/ |
||||||
|
private void showProgressNotification(int notifyId,String channelId,@DrawableRes int icon,CharSequence title,CharSequence content,int progress,int size){ |
||||||
|
NotificationCompat.Builder builder = buildNotification(channelId,icon,title,content,progress,size); |
||||||
|
builder.setAutoCancel(false); |
||||||
|
Notification notification = builder.build(); |
||||||
|
notification.flags = Notification.FLAG_AUTO_CANCEL | Notification.FLAG_ONLY_ALERT_ONCE; |
||||||
|
notifyNotification(notifyId,notification); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 显示下载完成时的通知(点击安装) |
||||||
|
* @param notifyId |
||||||
|
* @param channelId |
||||||
|
* @param icon |
||||||
|
* @param title |
||||||
|
* @param content |
||||||
|
* @param file |
||||||
|
*/ |
||||||
|
private void showFinishNotification(int notifyId,String channelId,@DrawableRes int icon,CharSequence title,CharSequence content,File file,String authority){ |
||||||
|
cancelNotification(notifyId); |
||||||
|
NotificationCompat.Builder builder = buildNotification(channelId,icon,title,content); |
||||||
|
builder.setAutoCancel(true); |
||||||
|
Intent intent = new Intent(Intent.ACTION_VIEW); |
||||||
|
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); |
||||||
|
intent.addCategory(Intent.CATEGORY_DEFAULT); |
||||||
|
Uri uriData; |
||||||
|
String type = "application/vnd.android.package-archive"; |
||||||
|
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){ |
||||||
|
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); |
||||||
|
uriData = FileProvider.getUriForFile(getContext(), authority, file); |
||||||
|
}else{ |
||||||
|
uriData = Uri.fromFile(file); |
||||||
|
} |
||||||
|
intent.setDataAndType(uriData, type); |
||||||
|
PendingIntent clickIntent = PendingIntent.getActivity(getContext(), notifyId,intent, PendingIntent.FLAG_UPDATE_CURRENT); |
||||||
|
builder.setContentIntent(clickIntent); |
||||||
|
Notification notification = builder.build(); |
||||||
|
notification.flags = Notification.FLAG_AUTO_CANCEL; |
||||||
|
notifyNotification(notifyId,notification); |
||||||
|
} |
||||||
|
|
||||||
|
private void showErrorNotification(int notifyId,String channelId,@DrawableRes int icon,CharSequence title,CharSequence content,boolean isReDownload,UpdateConfig config){ |
||||||
|
NotificationCompat.Builder builder = buildNotification(channelId,icon,title,content); |
||||||
|
builder.setAutoCancel(true); |
||||||
|
if(isReDownload){//重新下载
|
||||||
|
Intent intent = new Intent(getContext(),DownloadService.class); |
||||||
|
intent.putExtra(Constants.KEY_RE_DOWNLOAD,true); |
||||||
|
intent.putExtra(Constants.KEY_UPDATE_CONFIG,config); |
||||||
|
PendingIntent clickIntent = PendingIntent.getService(getContext(), notifyId,intent, PendingIntent.FLAG_UPDATE_CURRENT); |
||||||
|
builder.setContentIntent(clickIntent); |
||||||
|
}else{ |
||||||
|
PendingIntent clickIntent = PendingIntent.getService(getContext(), notifyId,new Intent(), PendingIntent.FLAG_UPDATE_CURRENT); |
||||||
|
builder.setContentIntent(clickIntent); |
||||||
|
} |
||||||
|
|
||||||
|
Notification notification = builder.build(); |
||||||
|
notification.flags = Notification.FLAG_AUTO_CANCEL; |
||||||
|
notifyNotification(notifyId,notification); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* 显示通知信息(非第一次) |
||||||
|
* @param notifyId |
||||||
|
* @param channelId |
||||||
|
* @param icon |
||||||
|
* @param title |
||||||
|
* @param content |
||||||
|
*/ |
||||||
|
private void showNotification(int notifyId,String channelId,@DrawableRes int icon,CharSequence title,CharSequence content,boolean isAutoCancel){ |
||||||
|
NotificationCompat.Builder builder = buildNotification(channelId,icon,title,content); |
||||||
|
builder.setAutoCancel(isAutoCancel); |
||||||
|
Notification notification = builder.build(); |
||||||
|
notification.flags = Notification.FLAG_AUTO_CANCEL; |
||||||
|
notifyNotification(notifyId,notification); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param notifyId |
||||||
|
*/ |
||||||
|
private void cancelNotification(int notifyId){ |
||||||
|
getNotificationManager().cancel(notifyId); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* 获取通知管理器 |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
private NotificationManager getNotificationManager(){ |
||||||
|
return (NotificationManager)getSystemService(Context.NOTIFICATION_SERVICE); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 创建一个通知渠道(兼容0以上版本) |
||||||
|
* @param channelId |
||||||
|
* @param channelName |
||||||
|
*/ |
||||||
|
@RequiresApi(api = Build.VERSION_CODES.O) |
||||||
|
private void createNotificationChannel(String channelId, String channelName){ |
||||||
|
NotificationChannel channel = new NotificationChannel(channelId,channelName, NotificationManager.IMPORTANCE_DEFAULT); |
||||||
|
getNotificationManager().createNotificationChannel(channel); |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 构建一个通知构建器 |
||||||
|
* @param channelId |
||||||
|
* @param icon |
||||||
|
* @param title |
||||||
|
* @param content |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
private NotificationCompat.Builder buildNotification(String channelId, @DrawableRes int icon,CharSequence title,CharSequence content){ |
||||||
|
return buildNotification(channelId,icon,title,content,Constants.NONE,Constants.NONE); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 构建一个通知构建器 |
||||||
|
* @param channelId |
||||||
|
* @param icon |
||||||
|
* @param title |
||||||
|
* @param content |
||||||
|
* @param progress |
||||||
|
* @param size |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
private NotificationCompat.Builder buildNotification(String channelId,@DrawableRes int icon,CharSequence title,CharSequence content,int progress,int size){ |
||||||
|
NotificationCompat.Builder builder = new NotificationCompat.Builder(getContext(),channelId); |
||||||
|
builder.setSmallIcon(icon); |
||||||
|
|
||||||
|
builder.setContentTitle(title); |
||||||
|
builder.setContentText(content); |
||||||
|
builder.setOngoing(true); |
||||||
|
|
||||||
|
if(progress!= Constants.NONE && size!=Constants.NONE){ |
||||||
|
builder.setProgress(size,progress,false); |
||||||
|
} |
||||||
|
|
||||||
|
return builder; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 更新通知栏 |
||||||
|
* @param id |
||||||
|
* @param notification |
||||||
|
*/ |
||||||
|
private void notifyNotification(int id, Notification notification){ |
||||||
|
getNotificationManager().notify(id,notification); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
//---------------------------------------- Binder
|
||||||
|
|
||||||
|
@Nullable |
||||||
|
@Override |
||||||
|
public IBinder onBind(Intent intent) { |
||||||
|
return mDownloadBinder; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 提供绑定服务的方式下载 |
||||||
|
*/ |
||||||
|
public class DownloadBinder extends Binder { |
||||||
|
|
||||||
|
public void start(UpdateConfig config){ |
||||||
|
start(config,null,null); |
||||||
|
} |
||||||
|
|
||||||
|
public void start(UpdateConfig config,UpdateCallback callback){ |
||||||
|
start(config,null,callback); |
||||||
|
} |
||||||
|
|
||||||
|
public void start(UpdateConfig config,IHttpManager httpManager,UpdateCallback callback){ |
||||||
|
startDownload(config,httpManager,callback); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
} |
@ -0,0 +1,98 @@ |
|||||||
|
package com.king.app.updater.util; |
||||||
|
|
||||||
|
import android.content.Context; |
||||||
|
import android.content.Intent; |
||||||
|
import android.content.pm.PackageInfo; |
||||||
|
import android.content.pm.PackageManager; |
||||||
|
import android.net.Uri; |
||||||
|
import android.os.Build; |
||||||
|
import android.support.v4.content.FileProvider; |
||||||
|
import android.text.TextUtils; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public enum AppUtils { |
||||||
|
|
||||||
|
INSTANCE; |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param context |
||||||
|
* @return AppName.apk |
||||||
|
*/ |
||||||
|
public String getAppFullName(Context context,String url,String defaultName){ |
||||||
|
if(url.endsWith(".apk")){ |
||||||
|
return url.substring(url.lastIndexOf("/") + 1); |
||||||
|
} |
||||||
|
|
||||||
|
String filename = getAppName(context); |
||||||
|
|
||||||
|
if(TextUtils.isEmpty(filename)){ |
||||||
|
filename = defaultName; |
||||||
|
} |
||||||
|
|
||||||
|
return String.format("%s.apk",filename); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public PackageInfo getPackageInfo(Context context) throws PackageManager.NameNotFoundException { |
||||||
|
PackageManager packageManager = context.getPackageManager(); |
||||||
|
PackageInfo packageInfo = packageManager.getPackageInfo( context.getPackageName(), 0); |
||||||
|
return packageInfo; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* 获取App的名称 |
||||||
|
*/ |
||||||
|
public String getAppName(Context context) { |
||||||
|
try{ |
||||||
|
|
||||||
|
int labelRes = getPackageInfo(context).applicationInfo.labelRes; |
||||||
|
return context.getResources().getString(labelRes); |
||||||
|
} catch (Exception e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
return null; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获取App的图标 |
||||||
|
* @param context |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public int getAppIcon(Context context){ |
||||||
|
try{ |
||||||
|
return getPackageInfo(context).applicationInfo.icon; |
||||||
|
} catch (Exception e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
|
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 安装apk |
||||||
|
* @param context |
||||||
|
* @param file |
||||||
|
*/ |
||||||
|
public void installApk(Context context,File file,String authority){ |
||||||
|
|
||||||
|
Intent intent = new Intent(Intent.ACTION_VIEW); |
||||||
|
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); |
||||||
|
intent.addCategory(Intent.CATEGORY_DEFAULT); |
||||||
|
Uri uriData = null; |
||||||
|
String type = "application/vnd.android.package-archive"; |
||||||
|
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){ |
||||||
|
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); |
||||||
|
uriData = FileProvider.getUriForFile(context, authority, file); |
||||||
|
}else{ |
||||||
|
uriData = Uri.fromFile(file); |
||||||
|
} |
||||||
|
intent.setDataAndType(uriData, type); |
||||||
|
context.startActivity(intent); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,36 @@ |
|||||||
|
package com.king.app.updater.util; |
||||||
|
|
||||||
|
import android.Manifest; |
||||||
|
import android.app.Activity; |
||||||
|
import android.content.pm.PackageManager; |
||||||
|
import android.support.annotation.NonNull; |
||||||
|
import android.support.v4.app.ActivityCompat; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public enum PermissionUtils { |
||||||
|
|
||||||
|
INSTANCE; |
||||||
|
|
||||||
|
private String[] PERMISSIONS_STORAGE = { |
||||||
|
Manifest.permission.READ_EXTERNAL_STORAGE, |
||||||
|
Manifest.permission.WRITE_EXTERNAL_STORAGE, |
||||||
|
Manifest.permission.READ_PHONE_STATE}; |
||||||
|
|
||||||
|
public boolean verifyReadAndWritePermissions(@NonNull Activity activity,int requestCode){ |
||||||
|
|
||||||
|
int readResult = checkPermission(activity,Manifest.permission.READ_EXTERNAL_STORAGE); |
||||||
|
int writeResult = checkPermission(activity,Manifest.permission.WRITE_EXTERNAL_STORAGE); |
||||||
|
int readPhoneState = checkPermission(activity,Manifest.permission.READ_PHONE_STATE); |
||||||
|
if( readResult != PackageManager.PERMISSION_GRANTED || writeResult != PackageManager.PERMISSION_GRANTED || readPhoneState != PackageManager.PERMISSION_GRANTED){ |
||||||
|
ActivityCompat.requestPermissions(activity,PERMISSIONS_STORAGE,requestCode); |
||||||
|
return false; |
||||||
|
} |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
public int checkPermission(@NonNull Activity activity,@NonNull String permission){ |
||||||
|
return ActivityCompat.checkSelfPermission(activity,permission); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,175 @@ |
|||||||
|
package com.king.app.updater.util; |
||||||
|
|
||||||
|
import android.content.Context; |
||||||
|
import android.support.annotation.RawRes; |
||||||
|
|
||||||
|
import java.io.InputStream; |
||||||
|
import java.security.KeyManagementException; |
||||||
|
import java.security.KeyStore; |
||||||
|
import java.security.NoSuchAlgorithmException; |
||||||
|
import java.security.SecureRandom; |
||||||
|
import java.security.cert.Certificate; |
||||||
|
import java.security.cert.CertificateException; |
||||||
|
import java.security.cert.CertificateFactory; |
||||||
|
import java.security.cert.X509Certificate; |
||||||
|
|
||||||
|
import javax.net.ssl.HostnameVerifier; |
||||||
|
import javax.net.ssl.SSLContext; |
||||||
|
import javax.net.ssl.SSLSession; |
||||||
|
import javax.net.ssl.SSLSocketFactory; |
||||||
|
import javax.net.ssl.TrustManager; |
||||||
|
import javax.net.ssl.TrustManagerFactory; |
||||||
|
import javax.net.ssl.X509TrustManager; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Jenly <a href="mailto:jenly1314@gmail.com">Jenly</a> |
||||||
|
*/ |
||||||
|
public class SSLSocketFactoryUtils { |
||||||
|
|
||||||
|
private SSLSocketFactoryUtils(){ |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
public static SSLSocketFactory createSSLSocketFactory() { |
||||||
|
SSLSocketFactory sslSocketFactory = null; |
||||||
|
try { |
||||||
|
SSLContext sslContext = SSLContext.getInstance("TLS"); |
||||||
|
sslContext.init(null, new TrustManager[]{createTrustAllManager()}, new SecureRandom()); |
||||||
|
sslSocketFactory = sslContext.getSocketFactory(); |
||||||
|
} catch (Exception e) { |
||||||
|
|
||||||
|
} |
||||||
|
return sslSocketFactory; |
||||||
|
} |
||||||
|
|
||||||
|
public static X509TrustManager createTrustAllManager() { |
||||||
|
X509TrustManager tm = null; |
||||||
|
try { |
||||||
|
tm = new X509TrustManager() { |
||||||
|
public void checkClientTrusted(X509Certificate[] chain, String authType) |
||||||
|
throws CertificateException { |
||||||
|
//do nothing
|
||||||
|
} |
||||||
|
|
||||||
|
public void checkServerTrusted(X509Certificate[] chain, String authType) |
||||||
|
throws CertificateException { |
||||||
|
//do nothing
|
||||||
|
} |
||||||
|
|
||||||
|
public X509Certificate[] getAcceptedIssuers() { |
||||||
|
return new X509Certificate[0]; |
||||||
|
} |
||||||
|
}; |
||||||
|
} catch (Exception e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
return tm; |
||||||
|
} |
||||||
|
|
||||||
|
public static TrustAllHostnameVerifier createTrustAllHostnameVerifier(){ |
||||||
|
return new TrustAllHostnameVerifier(); |
||||||
|
} |
||||||
|
|
||||||
|
public static class TrustAllHostnameVerifier implements HostnameVerifier { |
||||||
|
@Override |
||||||
|
public boolean verify(String hostname, SSLSession session) { |
||||||
|
return true; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param context |
||||||
|
* @param keyServerStoreID |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public static SSLSocketFactory createSSLSocketFactory(Context context,@RawRes int keyServerStoreID){ |
||||||
|
InputStream trustStream = context.getResources().openRawResource(keyServerStoreID); |
||||||
|
return createSSLSocketFactory(trustStream); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* @param certificates |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public static SSLSocketFactory createSSLSocketFactory(InputStream... certificates) { |
||||||
|
SSLSocketFactory mSSLSocketFactory = null; |
||||||
|
if(mSSLSocketFactory==null){ |
||||||
|
synchronized (SSLSocketFactoryUtils.class) { |
||||||
|
if(mSSLSocketFactory==null){ |
||||||
|
|
||||||
|
SSLContext sslContext = null; |
||||||
|
try { |
||||||
|
sslContext = SSLContext.getInstance("TLS"); |
||||||
|
} catch (NoSuchAlgorithmException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
return null; |
||||||
|
} |
||||||
|
//获得服务器端证书
|
||||||
|
TrustManager[] turstManager = getTurstManager(certificates); |
||||||
|
|
||||||
|
//初始化ssl证书库
|
||||||
|
try { |
||||||
|
sslContext.init(null,turstManager,new SecureRandom()); |
||||||
|
} catch (KeyManagementException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
|
||||||
|
//获得sslSocketFactory
|
||||||
|
mSSLSocketFactory = sslContext.getSocketFactory(); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return mSSLSocketFactory; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获得指定流中的服务器端证书库 |
||||||
|
* @param certificates |
||||||
|
* @return |
||||||
|
*/ |
||||||
|
public static TrustManager[] getTurstManager(InputStream... certificates) { |
||||||
|
try { |
||||||
|
CertificateFactory certificateFactory = CertificateFactory.getInstance("X.509"); |
||||||
|
KeyStore keyStore = KeyStore.getInstance(KeyStore.getDefaultType()); |
||||||
|
keyStore.load(null,null); |
||||||
|
int index = 0; |
||||||
|
for (InputStream certificate : certificates) { |
||||||
|
if (certificate == null) { |
||||||
|
continue; |
||||||
|
} |
||||||
|
Certificate certificate1; |
||||||
|
try { |
||||||
|
certificate1 = certificateFactory.generateCertificate(certificate); |
||||||
|
}finally { |
||||||
|
certificate.close(); |
||||||
|
} |
||||||
|
|
||||||
|
String certificateAlias = Integer.toString(index++); |
||||||
|
keyStore.setCertificateEntry(certificateAlias,certificate1); |
||||||
|
} |
||||||
|
|
||||||
|
TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory |
||||||
|
.getDefaultAlgorithm()); |
||||||
|
|
||||||
|
trustManagerFactory.init(keyStore); |
||||||
|
return trustManagerFactory.getTrustManagers(); |
||||||
|
|
||||||
|
} catch (Exception e) { |
||||||
|
e.printStackTrace(); |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
return getTurstAllManager(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获得信任所有服务器端证书库 |
||||||
|
* */ |
||||||
|
public static TrustManager[] getTurstAllManager() { |
||||||
|
return new X509TrustManager[] {createTrustAllManager()}; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
} |
@ -0,0 +1,18 @@ |
|||||||
|
<resources> |
||||||
|
<string name="app_name">AppUpdater</string> |
||||||
|
|
||||||
|
<string name="app_updater_start_notification_title">版本更新</string> |
||||||
|
<string name="app_updater_start_notification_content">正在获取下载数据…</string> |
||||||
|
|
||||||
|
<string name="app_updater_progress_notification_title">版本更新</string> |
||||||
|
<string name="app_updater_progress_notification_content">下载更新中…</string> |
||||||
|
|
||||||
|
<string name="app_updater_finish_notification_title">下载完成</string> |
||||||
|
<string name="app_updater_finish_notification_content">点击安装</string> |
||||||
|
|
||||||
|
<string name="app_updater_error_notification_title">下载失败</string> |
||||||
|
<string name="app_updater_error_notification_content">点击关闭通知</string> |
||||||
|
<string name="app_updater_error_notification_content_re_download">点击重新下载</string> |
||||||
|
|
||||||
|
|
||||||
|
</resources> |
@ -0,0 +1,6 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<paths> |
||||||
|
<external-path name="root_path" path="."/> |
||||||
|
<external-path name="app_updater_path" path=".AppUpdater"/> |
||||||
|
<cache-path name="data_path" path="data/data"/> |
||||||
|
</paths> |
@ -0,0 +1,17 @@ |
|||||||
|
package com.king.app.updater; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
|
||||||
|
import static org.junit.Assert.*; |
||||||
|
|
||||||
|
/** |
||||||
|
* Example local unit test, which will execute on the development machine (host). |
||||||
|
* |
||||||
|
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a> |
||||||
|
*/ |
||||||
|
public class ExampleUnitTest { |
||||||
|
@Test |
||||||
|
public void addition_isCorrect() { |
||||||
|
assertEquals(4, 2 + 2); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1 @@ |
|||||||
|
/build |
@ -0,0 +1,30 @@ |
|||||||
|
apply plugin: 'com.android.application' |
||||||
|
|
||||||
|
android { |
||||||
|
compileSdkVersion 27 |
||||||
|
defaultConfig { |
||||||
|
applicationId "com.king.appupdater" |
||||||
|
minSdkVersion 15 |
||||||
|
targetSdkVersion 27 |
||||||
|
versionCode 1 |
||||||
|
versionName "1.0" |
||||||
|
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" |
||||||
|
} |
||||||
|
buildTypes { |
||||||
|
release { |
||||||
|
minifyEnabled false |
||||||
|
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
dependencies { |
||||||
|
implementation fileTree(include: ['*.jar'], dir: 'libs') |
||||||
|
implementation 'com.android.support:appcompat-v7:27.1.1' |
||||||
|
implementation 'com.android.support.constraint:constraint-layout:1.1.2' |
||||||
|
testImplementation 'junit:junit:4.12' |
||||||
|
androidTestImplementation 'com.android.support.test:runner:1.0.2' |
||||||
|
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2' |
||||||
|
implementation project(':app-updater') |
||||||
|
implementation project(':app-dialog') |
||||||
|
} |
@ -0,0 +1,21 @@ |
|||||||
|
# Add project specific ProGuard rules here. |
||||||
|
# You can control the set of applied configuration files using the |
||||||
|
# proguardFiles setting in build.gradle. |
||||||
|
# |
||||||
|
# For more details, see |
||||||
|
# http://developer.android.com/guide/developing/tools/proguard.html |
||||||
|
|
||||||
|
# If your project uses WebView with JS, uncomment the following |
||||||
|
# and specify the fully qualified class name to the JavaScript interface |
||||||
|
# class: |
||||||
|
#-keepclassmembers class fqcn.of.javascript.interface.for.webview { |
||||||
|
# public *; |
||||||
|
#} |
||||||
|
|
||||||
|
# Uncomment this to preserve the line number information for |
||||||
|
# debugging stack traces. |
||||||
|
#-keepattributes SourceFile,LineNumberTable |
||||||
|
|
||||||
|
# If you keep the line number information, uncomment this to |
||||||
|
# hide the original source file name. |
||||||
|
#-renamesourcefileattribute SourceFile |
@ -0,0 +1 @@ |
|||||||
|
[{"outputType":{"type":"APK"},"apkInfo":{"type":"MAIN","splits":[],"versionCode":1,"versionName":"1.0","enabled":true,"outputFile":"app-release.apk","fullName":"release","baseName":"release"},"path":"app-release.apk","properties":{}}] |
@ -0,0 +1,26 @@ |
|||||||
|
package com.king.appupdater; |
||||||
|
|
||||||
|
import android.content.Context; |
||||||
|
import android.support.test.InstrumentationRegistry; |
||||||
|
import android.support.test.runner.AndroidJUnit4; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
import org.junit.runner.RunWith; |
||||||
|
|
||||||
|
import static org.junit.Assert.*; |
||||||
|
|
||||||
|
/** |
||||||
|
* Instrumented test, which will execute on an Android device. |
||||||
|
* |
||||||
|
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a> |
||||||
|
*/ |
||||||
|
@RunWith(AndroidJUnit4.class) |
||||||
|
public class ExampleInstrumentedTest { |
||||||
|
@Test |
||||||
|
public void useAppContext() { |
||||||
|
// Context of the app under test.
|
||||||
|
Context appContext = InstrumentationRegistry.getTargetContext(); |
||||||
|
|
||||||
|
assertEquals("com.king.appupdater", appContext.getPackageName()); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,27 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
package="com.king.appupdater"> |
||||||
|
|
||||||
|
<uses-permission android:name="android.permission.INTERNET" /> |
||||||
|
|
||||||
|
<application |
||||||
|
android:allowBackup="true" |
||||||
|
android:icon="@mipmap/ic_launcher" |
||||||
|
android:label="@string/app_name" |
||||||
|
android:roundIcon="@mipmap/ic_launcher" |
||||||
|
android:supportsRtl="true" |
||||||
|
android:theme="@style/AppTheme"> |
||||||
|
<activity android:name=".MainActivity" |
||||||
|
android:screenOrientation="portrait"> |
||||||
|
<intent-filter> |
||||||
|
<action android:name="android.intent.action.MAIN" /> |
||||||
|
|
||||||
|
<category android:name="android.intent.category.LAUNCHER" /> |
||||||
|
</intent-filter> |
||||||
|
</activity> |
||||||
|
|
||||||
|
</application> |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</manifest> |
After Width: | Height: | Size: 19 KiB |
@ -0,0 +1,274 @@ |
|||||||
|
package com.king.appupdater; |
||||||
|
|
||||||
|
import android.content.Context; |
||||||
|
import android.content.DialogInterface; |
||||||
|
import android.os.Environment; |
||||||
|
import android.support.v7.app.AlertDialog; |
||||||
|
import android.support.v7.app.AppCompatActivity; |
||||||
|
import android.os.Bundle; |
||||||
|
import android.view.LayoutInflater; |
||||||
|
import android.view.View; |
||||||
|
import android.widget.Button; |
||||||
|
import android.widget.ProgressBar; |
||||||
|
import android.widget.TextView; |
||||||
|
import android.widget.Toast; |
||||||
|
|
||||||
|
import com.king.app.dialog.AppDialog; |
||||||
|
import com.king.app.dialog.AppDialogConfig; |
||||||
|
import com.king.app.updater.AppUpdater; |
||||||
|
import com.king.app.updater.UpdateConfig; |
||||||
|
import com.king.app.updater.callback.AppUpdateCallback; |
||||||
|
import com.king.app.updater.callback.UpdateCallback; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
|
||||||
|
public class MainActivity extends AppCompatActivity { |
||||||
|
|
||||||
|
private final Object mLock = new Object(); |
||||||
|
|
||||||
|
public final String TAG = this.getClass().getSimpleName(); |
||||||
|
|
||||||
|
private String mUrl = "https://raw.githubusercontent.com/jenly1314/AppUpdater/master/app/release/app-release.apk"; |
||||||
|
// private String mUrl = "https://pro-app-qn.fir.im/1ddfe25998acd3d861d746101e6e079e1611b666.apk?attname=app-release.apk_1.2.apk";
|
||||||
|
|
||||||
|
private ProgressBar progressBar; |
||||||
|
|
||||||
|
private Toast toast; |
||||||
|
|
||||||
|
@Override |
||||||
|
protected void onCreate(Bundle savedInstanceState) { |
||||||
|
super.onCreate(savedInstanceState); |
||||||
|
setContentView(R.layout.activity_main); |
||||||
|
progressBar = findViewById(R.id.progressBar); |
||||||
|
progressBar.setVisibility(View.INVISIBLE); |
||||||
|
} |
||||||
|
|
||||||
|
public Context getContext(){ |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
public void showToast(String text){ |
||||||
|
if(toast == null){ |
||||||
|
synchronized (mLock){ |
||||||
|
if(toast == null){ |
||||||
|
toast = Toast.makeText(getContext(),text,Toast.LENGTH_SHORT); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
toast.setText(text); |
||||||
|
toast.show(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 简单一键后台升级 |
||||||
|
*/ |
||||||
|
private void clickBtn1(){ |
||||||
|
new AppUpdater(getContext(),mUrl).start(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 一键下载并监听 |
||||||
|
*/ |
||||||
|
private void clickBtn2(){ |
||||||
|
UpdateConfig config = new UpdateConfig(); |
||||||
|
config.setUrl(mUrl); |
||||||
|
new AppUpdater(getContext(),config) |
||||||
|
.setUpdateCallback(new UpdateCallback() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onDownloading(boolean isDownloading) { |
||||||
|
if(isDownloading){ |
||||||
|
showToast("已经在下载中,请勿重复下载。"); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onStart(String url) { |
||||||
|
progressBar.setProgress(0); |
||||||
|
progressBar.setVisibility(View.VISIBLE); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onProgress(int progress, int total, boolean isChange) { |
||||||
|
if(isChange){ |
||||||
|
progressBar.setMax(total); |
||||||
|
progressBar.setProgress(progress); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onFinish(File file) { |
||||||
|
progressBar.setVisibility(View.INVISIBLE); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onError(Exception e) { |
||||||
|
progressBar.setVisibility(View.INVISIBLE); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onCancel() { |
||||||
|
progressBar.setVisibility(View.INVISIBLE); |
||||||
|
} |
||||||
|
}) |
||||||
|
.start(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 系统弹框升级 |
||||||
|
*/ |
||||||
|
private void clickBtn3(){ |
||||||
|
new AlertDialog.Builder(this) |
||||||
|
.setTitle("发现新版本") |
||||||
|
.setMessage("1、新增某某功能、\n2、修改某某问题、\n3、优化某某BUG、") |
||||||
|
.setPositiveButton("升级", new DialogInterface.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(DialogInterface dialog, int which) { |
||||||
|
new AppUpdater.Builder() |
||||||
|
.serUrl(mUrl) |
||||||
|
.build(getContext()) |
||||||
|
.setUpdateCallback(new AppUpdateCallback() { |
||||||
|
@Override |
||||||
|
public void onProgress(int progress, int total, boolean isChange) { |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onFinish(File file) { |
||||||
|
showToast("下载完成"); |
||||||
|
} |
||||||
|
}) |
||||||
|
.start(); |
||||||
|
} |
||||||
|
}).show(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 简单弹框升级 |
||||||
|
*/ |
||||||
|
private void clickBtn4(){ |
||||||
|
AppDialogConfig config = new AppDialogConfig(); |
||||||
|
config.setTitle("简单弹框升级") |
||||||
|
.setOk("升级") |
||||||
|
.setContent("1、新增某某功能、\n2、修改某某问题、\n3、优化某某BUG、") |
||||||
|
.setOnClickOk(new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
new AppUpdater.Builder() |
||||||
|
.serUrl(mUrl) |
||||||
|
.setFilename("AppUpdater.apk") |
||||||
|
.build(getContext()) |
||||||
|
.start(); |
||||||
|
AppDialog.INSTANCE.dismissDialog(); |
||||||
|
} |
||||||
|
}); |
||||||
|
AppDialog.INSTANCE.showDialog(getContext(),config); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 简单自定义弹框升级 |
||||||
|
*/ |
||||||
|
private void clickBtn5(){ |
||||||
|
AppDialogConfig config = new AppDialogConfig(); |
||||||
|
config.setLayoutId(R.layout.dialog) |
||||||
|
.setOk("升级") |
||||||
|
.setHideCancel(true) |
||||||
|
.setTitle("简单自定义弹框升级") |
||||||
|
.setContent("1、新增某某功能、\n2、修改某某问题、\n3、优化某某BUG、") |
||||||
|
.setOnClickOk(new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
new AppUpdater.Builder() |
||||||
|
.serUrl(mUrl) |
||||||
|
.setFilename(Environment.getExternalStorageDirectory() + "/.AppUpdater") |
||||||
|
.setFilename("AppUpdater.apk") |
||||||
|
.build(getContext()) |
||||||
|
.start(); |
||||||
|
AppDialog.INSTANCE.dismissDialog(); |
||||||
|
} |
||||||
|
}); |
||||||
|
AppDialog.INSTANCE.showDialog(getContext(),AppDialog.INSTANCE.createAppDialogView(getContext(),config),false); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 自定义弹框升级 |
||||||
|
*/ |
||||||
|
private void clickBtn6(){ |
||||||
|
View view = LayoutInflater.from(getContext()).inflate(R.layout.dialog_custom,null); |
||||||
|
|
||||||
|
TextView tvTitle = view.findViewById(R.id.tvTitle); |
||||||
|
tvTitle.setText("自定义弹框升级"); |
||||||
|
TextView tvContent = view.findViewById(R.id.tvContent); |
||||||
|
tvContent.setText("1、新增某某功能、\n2、修改某某问题、\n3、优化某某BUG、"); |
||||||
|
|
||||||
|
Button btnCancel = view.findViewById(R.id.btnCancel); |
||||||
|
btnCancel.setOnClickListener(new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
AppDialog.INSTANCE.dismissDialog(); |
||||||
|
} |
||||||
|
}); |
||||||
|
Button btnOK = view.findViewById(R.id.btnOK); |
||||||
|
btnOK.setOnClickListener(new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
new AppUpdater.Builder() |
||||||
|
.serUrl(mUrl) |
||||||
|
.setFilename("AppUpdater.apk") |
||||||
|
.build(getContext()) |
||||||
|
.start(); |
||||||
|
AppDialog.INSTANCE.dismissDialog(); |
||||||
|
} |
||||||
|
}); |
||||||
|
|
||||||
|
AppDialog.INSTANCE.showDialog(getContext(),view); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 简单DialogFragment升级 |
||||||
|
*/ |
||||||
|
private void clickBtn7(){ |
||||||
|
AppDialogConfig config = new AppDialogConfig(); |
||||||
|
config.setTitle("简单DialogFragment升级") |
||||||
|
.setOk("升级") |
||||||
|
.setContent("1、新增某某功能、\n2、修改某某问题、\n3、优化某某BUG、") |
||||||
|
.setOnClickOk(new View.OnClickListener() { |
||||||
|
@Override |
||||||
|
public void onClick(View v) { |
||||||
|
new AppUpdater.Builder() |
||||||
|
.serUrl(mUrl) |
||||||
|
.setFilename("AppUpdater.apk") |
||||||
|
.build(getContext()) |
||||||
|
.start(); |
||||||
|
AppDialog.INSTANCE.dismissDialogFragment(getSupportFragmentManager()); |
||||||
|
} |
||||||
|
}); |
||||||
|
AppDialog.INSTANCE.showDialogFragment(getSupportFragmentManager(),config); |
||||||
|
} |
||||||
|
|
||||||
|
public void OnClick(View v){ |
||||||
|
switch (v.getId()){ |
||||||
|
case R.id.btn1: |
||||||
|
clickBtn1(); |
||||||
|
break; |
||||||
|
case R.id.btn2: |
||||||
|
clickBtn2(); |
||||||
|
break; |
||||||
|
case R.id.btn3: |
||||||
|
clickBtn3(); |
||||||
|
break; |
||||||
|
case R.id.btn4: |
||||||
|
clickBtn4(); |
||||||
|
break; |
||||||
|
case R.id.btn5: |
||||||
|
clickBtn5(); |
||||||
|
break; |
||||||
|
case R.id.btn6: |
||||||
|
clickBtn6(); |
||||||
|
break; |
||||||
|
case R.id.btn7: |
||||||
|
clickBtn7(); |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,34 @@ |
|||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
xmlns:aapt="http://schemas.android.com/aapt" |
||||||
|
android:width="108dp" |
||||||
|
android:height="108dp" |
||||||
|
android:viewportHeight="108" |
||||||
|
android:viewportWidth="108"> |
||||||
|
<path |
||||||
|
android:fillType="evenOdd" |
||||||
|
android:pathData="M32,64C32,64 38.39,52.99 44.13,50.95C51.37,48.37 70.14,49.57 70.14,49.57L108.26,87.69L108,109.01L75.97,107.97L32,64Z" |
||||||
|
android:strokeColor="#00000000" |
||||||
|
android:strokeWidth="1"> |
||||||
|
<aapt:attr name="android:fillColor"> |
||||||
|
<gradient |
||||||
|
android:endX="78.5885" |
||||||
|
android:endY="90.9159" |
||||||
|
android:startX="48.7653" |
||||||
|
android:startY="61.0927" |
||||||
|
android:type="linear"> |
||||||
|
<item |
||||||
|
android:color="#ffffff" |
||||||
|
android:offset="0.0" /> |
||||||
|
<item |
||||||
|
android:color="#ffffff" |
||||||
|
android:offset="1.0" /> |
||||||
|
</gradient> |
||||||
|
</aapt:attr> |
||||||
|
</path> |
||||||
|
<path |
||||||
|
android:fillColor="#FFFFFF" |
||||||
|
android:fillType="nonZero" |
||||||
|
android:pathData="M66.94,46.02L66.94,46.02C72.44,50.07 76,56.61 76,64L32,64C32,56.61 35.56,50.11 40.98,46.06L36.18,41.19C35.45,40.45 35.45,39.3 36.18,38.56C36.91,37.81 38.05,37.81 38.78,38.56L44.25,44.05C47.18,42.57 50.48,41.71 54,41.71C57.48,41.71 60.78,42.57 63.68,44.05L69.11,38.56C69.84,37.81 70.98,37.81 71.71,38.56C72.44,39.3 72.44,40.45 71.71,41.19L66.94,46.02ZM62.94,56.92C64.08,56.92 65,56.01 65,54.88C65,53.76 64.08,52.85 62.94,52.85C61.8,52.85 60.88,53.76 60.88,54.88C60.88,56.01 61.8,56.92 62.94,56.92ZM45.06,56.92C46.2,56.92 47.13,56.01 47.13,54.88C47.13,53.76 46.2,52.85 45.06,52.85C43.92,52.85 43,53.76 43,54.88C43,56.01 43.92,56.92 45.06,56.92Z" |
||||||
|
android:strokeColor="#00000000" |
||||||
|
android:strokeWidth="1" /> |
||||||
|
</vector> |
@ -0,0 +1,12 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<shape xmlns:android="http://schemas.android.com/apk/res/android" > |
||||||
|
|
||||||
|
<solid android:color="#F5F5F5" /> |
||||||
|
|
||||||
|
<corners |
||||||
|
android:bottomLeftRadius="0dp" |
||||||
|
android:bottomRightRadius="0dp" |
||||||
|
android:topLeftRadius="10dp" |
||||||
|
android:topRightRadius="10dp" /> |
||||||
|
|
||||||
|
</shape> |
@ -0,0 +1,170 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
android:width="108dp" |
||||||
|
android:height="108dp" |
||||||
|
android:viewportHeight="108" |
||||||
|
android:viewportWidth="108"> |
||||||
|
<path |
||||||
|
android:fillColor="#ffffff" |
||||||
|
android:pathData="M0,0h108v108h-108z" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M9,0L9,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M19,0L19,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M29,0L29,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M39,0L39,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M49,0L49,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M59,0L59,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M69,0L69,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M79,0L79,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M89,0L89,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M99,0L99,108" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,9L108,9" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,19L108,19" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,29L108,29" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,39L108,39" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,49L108,49" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,59L108,59" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,69L108,69" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,79L108,79" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,89L108,89" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M0,99L108,99" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M19,29L89,29" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M19,39L89,39" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M19,49L89,49" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M19,59L89,59" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M19,69L89,69" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M19,79L89,79" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M29,19L29,89" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M39,19L39,89" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M49,19L49,89" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M59,19L59,89" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M69,19L69,89" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
<path |
||||||
|
android:fillColor="#00000000" |
||||||
|
android:pathData="M79,19L79,89" |
||||||
|
android:strokeColor="#33FFFFFF" |
||||||
|
android:strokeWidth="0.8" /> |
||||||
|
</vector> |
@ -0,0 +1,107 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<android.support.constraint.ConstraintLayout 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" |
||||||
|
tools:context=".MainActivity"> |
||||||
|
|
||||||
|
<ProgressBar |
||||||
|
android:id="@+id/progressBar" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_margin="20dp" |
||||||
|
app:layout_constraintTop_toBottomOf="parent" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
app:layout_constraintTop_toTopOf="parent" |
||||||
|
style="@style/Widget.AppCompat.ProgressBar.Horizontal"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btn1" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_marginLeft="20dp" |
||||||
|
android:layout_marginRight="20dp" |
||||||
|
android:layout_marginTop="40dp" |
||||||
|
android:text="简单一键后台升级" |
||||||
|
app:layout_constraintTop_toBottomOf="@+id/progressBar" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
app:layout_constraintTop_toTopOf="parent" |
||||||
|
style="@style/OnClick"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btn2" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_marginLeft="20dp" |
||||||
|
android:layout_marginRight="20dp" |
||||||
|
android:layout_marginTop="10dp" |
||||||
|
android:text="一键监听进度后台升级" |
||||||
|
app:layout_constraintTop_toBottomOf="@+id/btn1" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
style="@style/OnClick"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btn3" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_marginLeft="20dp" |
||||||
|
android:layout_marginRight="20dp" |
||||||
|
android:layout_marginTop="10dp" |
||||||
|
android:text="系统弹框升级" |
||||||
|
app:layout_constraintTop_toBottomOf="@+id/btn2" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
style="@style/OnClick"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btn4" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_marginLeft="20dp" |
||||||
|
android:layout_marginRight="20dp" |
||||||
|
android:layout_marginTop="10dp" |
||||||
|
android:text="简单弹框升级" |
||||||
|
app:layout_constraintTop_toBottomOf="@+id/btn3" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
style="@style/OnClick"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btn5" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_marginLeft="20dp" |
||||||
|
android:layout_marginRight="20dp" |
||||||
|
android:layout_marginTop="10dp" |
||||||
|
android:text="简单自定义弹框升级" |
||||||
|
app:layout_constraintTop_toBottomOf="@+id/btn4" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
style="@style/OnClick"/> |
||||||
|
|
||||||
|
<Button |
||||||
|
android:id="@+id/btn6" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_marginLeft="20dp" |
||||||
|
android:layout_marginRight="20dp" |
||||||
|
android:layout_marginTop="10dp" |
||||||
|
android:text="自定义弹框升级" |
||||||
|
app:layout_constraintTop_toBottomOf="@+id/btn5" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
style="@style/OnClick"/> |
||||||
|
|
||||||
|
<Button |
||||||
|
android:id="@+id/btn7" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_marginLeft="20dp" |
||||||
|
android:layout_marginRight="20dp" |
||||||
|
android:layout_marginTop="10dp" |
||||||
|
android:text="简单DialogFragment升级" |
||||||
|
app:layout_constraintTop_toBottomOf="@+id/btn6" |
||||||
|
app:layout_constraintLeft_toLeftOf="parent" |
||||||
|
app:layout_constraintRight_toRightOf="parent" |
||||||
|
style="@style/OnClick"/> |
||||||
|
|
||||||
|
</android.support.constraint.ConstraintLayout> |
@ -0,0 +1,60 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_gravity="center" |
||||||
|
android:layout_margin="30dp" |
||||||
|
android:background="@drawable/app_dialog_bg" |
||||||
|
android:orientation="vertical"> |
||||||
|
<TextView |
||||||
|
android:id="@+id/tvDialogTitle" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:gravity="center" |
||||||
|
android:minHeight="40dp" |
||||||
|
android:padding="6dp" |
||||||
|
android:lines="1" |
||||||
|
android:textSize="16sp" |
||||||
|
android:background="@drawable/dialog_title_bg" |
||||||
|
android:text="@string/app_dialog_title"/> |
||||||
|
<TextView |
||||||
|
android:id="@+id/tvDialogContent" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:padding="10dp" |
||||||
|
android:layout_marginBottom="6dp" |
||||||
|
android:lineSpacingMultiplier="1" /> |
||||||
|
<include layout="@layout/app_dialog_line_h"/> |
||||||
|
<LinearLayout |
||||||
|
android:orientation="horizontal" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:gravity="center_horizontal"> |
||||||
|
<Button |
||||||
|
android:id="@+id/btnDialogCancel" |
||||||
|
android:layout_width="0dp" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:paddingTop="10dp" |
||||||
|
android:paddingBottom="10dp" |
||||||
|
android:layout_weight="1" |
||||||
|
android:text="@string/app_dialog_cancel" |
||||||
|
android:textSize="18sp" |
||||||
|
android:textColor="@color/app_dialog_button_color_selector" |
||||||
|
android:background="?android:attr/selectableItemBackground"/> |
||||||
|
<include |
||||||
|
android:id="@+id/line" |
||||||
|
layout="@layout/app_dialog_line_v"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btnDialogOK" |
||||||
|
android:layout_width="0dp" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:paddingTop="10dp" |
||||||
|
android:paddingBottom="10dp" |
||||||
|
android:layout_weight="1" |
||||||
|
android:text="@string/app_dialog_ok" |
||||||
|
android:textSize="18sp" |
||||||
|
android:textColor="@color/app_dialog_button_color_selector" |
||||||
|
android:background="?android:attr/selectableItemBackground"/> |
||||||
|
</LinearLayout> |
||||||
|
|
||||||
|
</LinearLayout> |
@ -0,0 +1,56 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:layout_gravity="center" |
||||||
|
android:layout_margin="30dp" |
||||||
|
android:background="@android:color/white" |
||||||
|
android:orientation="vertical"> |
||||||
|
<TextView |
||||||
|
android:id="@+id/tvTitle" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:gravity="center" |
||||||
|
android:minHeight="40dp" |
||||||
|
android:padding="6dp" |
||||||
|
android:lines="1" |
||||||
|
android:textSize="16sp" |
||||||
|
android:background="#F5F5F5" /> |
||||||
|
<TextView |
||||||
|
android:id="@+id/tvContent" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:padding="10dp" |
||||||
|
android:layout_marginBottom="6dp" |
||||||
|
android:lineSpacingMultiplier="1" /> |
||||||
|
<include layout="@layout/app_dialog_line_h"/> |
||||||
|
<LinearLayout |
||||||
|
android:orientation="horizontal" |
||||||
|
android:layout_width="match_parent" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:gravity="center_horizontal"> |
||||||
|
<Button |
||||||
|
android:id="@+id/btnCancel" |
||||||
|
android:layout_width="0dp" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:paddingTop="10dp" |
||||||
|
android:paddingBottom="10dp" |
||||||
|
android:layout_weight="1" |
||||||
|
android:text="@string/app_dialog_cancel" |
||||||
|
android:textSize="16sp" |
||||||
|
android:textColor="@color/app_dialog_button_color_selector" |
||||||
|
android:background="?android:attr/selectableItemBackground"/> |
||||||
|
<Button |
||||||
|
android:id="@+id/btnOK" |
||||||
|
android:layout_width="0dp" |
||||||
|
android:layout_height="wrap_content" |
||||||
|
android:paddingTop="10dp" |
||||||
|
android:paddingBottom="10dp" |
||||||
|
android:layout_weight="1" |
||||||
|
android:text="升级" |
||||||
|
android:textSize="16sp" |
||||||
|
android:textColor="@color/app_dialog_button_color_selector" |
||||||
|
android:background="?android:attr/selectableItemBackground"/> |
||||||
|
</LinearLayout> |
||||||
|
|
||||||
|
</LinearLayout> |
@ -0,0 +1,5 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> |
||||||
|
<background android:drawable="@color/ic_launcher_background"/> |
||||||
|
<foreground android:drawable="@mipmap/ic_launcher_foreground"/> |
||||||
|
</adaptive-icon> |
@ -0,0 +1,5 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> |
||||||
|
<background android:drawable="@color/ic_launcher_background"/> |
||||||
|
<foreground android:drawable="@mipmap/ic_launcher_foreground"/> |
||||||
|
</adaptive-icon> |
After Width: | Height: | Size: 1.9 KiB |
After Width: | Height: | Size: 2.1 KiB |
After Width: | Height: | Size: 3.7 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.3 KiB |
After Width: | Height: | Size: 2.3 KiB |
After Width: | Height: | Size: 2.5 KiB |
After Width: | Height: | Size: 2.9 KiB |
After Width: | Height: | Size: 5.1 KiB |
After Width: | Height: | Size: 3.9 KiB |
After Width: | Height: | Size: 4.7 KiB |
After Width: | Height: | Size: 8.1 KiB |
After Width: | Height: | Size: 5.3 KiB |
After Width: | Height: | Size: 6.9 KiB |
After Width: | Height: | Size: 12 KiB |
@ -0,0 +1,10 @@ |
|||||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||||
|
<resources> |
||||||
|
<color name="colorPrimary">#3F51B5</color> |
||||||
|
<color name="colorPrimaryDark">#303F9F</color> |
||||||
|
<color name="colorAccent">#FF4081</color> |
||||||
|
|
||||||
|
<color name="ic_launcher_background">#FFFFFF</color> |
||||||
|
|
||||||
|
|
||||||
|
</resources> |
@ -0,0 +1,4 @@ |
|||||||
|
<resources> |
||||||
|
<string name="app_name">AppUpdater</string> |
||||||
|
|
||||||
|
</resources> |
@ -0,0 +1,16 @@ |
|||||||
|
<resources> |
||||||
|
|
||||||
|
<!-- Base application theme. --> |
||||||
|
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar"> |
||||||
|
<!-- Customize your theme here. --> |
||||||
|
<item name="colorPrimary">@color/colorPrimary</item> |
||||||
|
<item name="colorPrimaryDark">@color/colorPrimaryDark</item> |
||||||
|
<item name="colorAccent">@color/colorAccent</item> |
||||||
|
<item name="android:textAllCaps">false</item> |
||||||
|
</style> |
||||||
|
|
||||||
|
<style name="OnClick"> |
||||||
|
<item name="android:clickable">true</item> |
||||||
|
<item name="android:onClick">OnClick</item> |
||||||
|
</style> |
||||||
|
</resources> |
@ -0,0 +1,17 @@ |
|||||||
|
package com.king.appupdater; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
|
||||||
|
import static org.junit.Assert.*; |
||||||
|
|
||||||
|
/** |
||||||
|
* Example local unit test, which will execute on the development machine (host). |
||||||
|
* |
||||||
|
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a> |
||||||
|
*/ |
||||||
|
public class ExampleUnitTest { |
||||||
|
@Test |
||||||
|
public void addition_isCorrect() { |
||||||
|
assertEquals(4, 2 + 2); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,37 @@ |
|||||||
|
// Top-level build file where you can add configuration options common to all sub-projects/modules. |
||||||
|
|
||||||
|
buildscript { |
||||||
|
|
||||||
|
repositories { |
||||||
|
google() |
||||||
|
jcenter() |
||||||
|
} |
||||||
|
dependencies { |
||||||
|
classpath 'com.android.tools.build:gradle:3.1.2' |
||||||
|
|
||||||
|
classpath 'com.novoda:bintray-release:0.8.1' |
||||||
|
// NOTE: Do not place your application dependencies here; they belong |
||||||
|
// in the individual module build.gradle files |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
allprojects { |
||||||
|
repositories { |
||||||
|
google() |
||||||
|
jcenter() |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
task clean(type: Delete) { |
||||||
|
delete rootProject.buildDir |
||||||
|
} |
||||||
|
|
||||||
|
allprojects { |
||||||
|
tasks.withType(Javadoc) { |
||||||
|
options{ |
||||||
|
encoding "UTF-8" |
||||||
|
charSet 'UTF-8' |
||||||
|
links "http://docs.oracle.com/javase/8/docs/api" |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,13 @@ |
|||||||
|
# Project-wide Gradle settings. |
||||||
|
# IDE (e.g. Android Studio) users: |
||||||
|
# Gradle settings configured through the IDE *will override* |
||||||
|
# any settings specified in this file. |
||||||
|
# For more details on how to configure your build environment visit |
||||||
|
# http://www.gradle.org/docs/current/userguide/build_environment.html |
||||||
|
# Specifies the JVM arguments used for the daemon process. |
||||||
|
# The setting is particularly useful for tweaking memory settings. |
||||||
|
org.gradle.jvmargs=-Xmx1536m |
||||||
|
# When configured, Gradle will run in incubating parallel mode. |
||||||
|
# This option should only be used with decoupled projects. More details, visit |
||||||
|
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects |
||||||
|
# org.gradle.parallel=true |
@ -0,0 +1,6 @@ |
|||||||
|
#Mon Jun 25 18:24:14 CST 2018 |
||||||
|
distributionBase=GRADLE_USER_HOME |
||||||
|
distributionPath=wrapper/dists |
||||||
|
zipStoreBase=GRADLE_USER_HOME |
||||||
|
zipStorePath=wrapper/dists |
||||||
|
distributionUrl=https\://services.gradle.org/distributions/gradle-4.4-all.zip |
@ -0,0 +1,172 @@ |
|||||||
|
#!/usr/bin/env sh |
||||||
|
|
||||||
|
############################################################################## |
||||||
|
## |
||||||
|
## Gradle start up script for UN*X |
||||||
|
## |
||||||
|
############################################################################## |
||||||
|
|
||||||
|
# Attempt to set APP_HOME |
||||||
|
# Resolve links: $0 may be a link |
||||||
|
PRG="$0" |
||||||
|
# Need this for relative symlinks. |
||||||
|
while [ -h "$PRG" ] ; do |
||||||
|
ls=`ls -ld "$PRG"` |
||||||
|
link=`expr "$ls" : '.*-> \(.*\)$'` |
||||||
|
if expr "$link" : '/.*' > /dev/null; then |
||||||
|
PRG="$link" |
||||||
|
else |
||||||
|
PRG=`dirname "$PRG"`"/$link" |
||||||
|
fi |
||||||
|
done |
||||||
|
SAVED="`pwd`" |
||||||
|
cd "`dirname \"$PRG\"`/" >/dev/null |
||||||
|
APP_HOME="`pwd -P`" |
||||||
|
cd "$SAVED" >/dev/null |
||||||
|
|
||||||
|
APP_NAME="Gradle" |
||||||
|
APP_BASE_NAME=`basename "$0"` |
||||||
|
|
||||||
|
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. |
||||||
|
DEFAULT_JVM_OPTS="" |
||||||
|
|
||||||
|
# Use the maximum available, or set MAX_FD != -1 to use that value. |
||||||
|
MAX_FD="maximum" |
||||||
|
|
||||||
|
warn () { |
||||||
|
echo "$*" |
||||||
|
} |
||||||
|
|
||||||
|
die () { |
||||||
|
echo |
||||||
|
echo "$*" |
||||||
|
echo |
||||||
|
exit 1 |
||||||
|
} |
||||||
|
|
||||||
|
# OS specific support (must be 'true' or 'false'). |
||||||
|
cygwin=false |
||||||
|
msys=false |
||||||
|
darwin=false |
||||||
|
nonstop=false |
||||||
|
case "`uname`" in |
||||||
|
CYGWIN* ) |
||||||
|
cygwin=true |
||||||
|
;; |
||||||
|
Darwin* ) |
||||||
|
darwin=true |
||||||
|
;; |
||||||
|
MINGW* ) |
||||||
|
msys=true |
||||||
|
;; |
||||||
|
NONSTOP* ) |
||||||
|
nonstop=true |
||||||
|
;; |
||||||
|
esac |
||||||
|
|
||||||
|
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar |
||||||
|
|
||||||
|
# Determine the Java command to use to start the JVM. |
||||||
|
if [ -n "$JAVA_HOME" ] ; then |
||||||
|
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then |
||||||
|
# IBM's JDK on AIX uses strange locations for the executables |
||||||
|
JAVACMD="$JAVA_HOME/jre/sh/java" |
||||||
|
else |
||||||
|
JAVACMD="$JAVA_HOME/bin/java" |
||||||
|
fi |
||||||
|
if [ ! -x "$JAVACMD" ] ; then |
||||||
|
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME |
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the |
||||||
|
location of your Java installation." |
||||||
|
fi |
||||||
|
else |
||||||
|
JAVACMD="java" |
||||||
|
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. |
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the |
||||||
|
location of your Java installation." |
||||||
|
fi |
||||||
|
|
||||||
|
# Increase the maximum file descriptors if we can. |
||||||
|
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then |
||||||
|
MAX_FD_LIMIT=`ulimit -H -n` |
||||||
|
if [ $? -eq 0 ] ; then |
||||||
|
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then |
||||||
|
MAX_FD="$MAX_FD_LIMIT" |
||||||
|
fi |
||||||
|
ulimit -n $MAX_FD |
||||||
|
if [ $? -ne 0 ] ; then |
||||||
|
warn "Could not set maximum file descriptor limit: $MAX_FD" |
||||||
|
fi |
||||||
|
else |
||||||
|
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" |
||||||
|
fi |
||||||
|
fi |
||||||
|
|
||||||
|
# For Darwin, add options to specify how the application appears in the dock |
||||||
|
if $darwin; then |
||||||
|
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" |
||||||
|
fi |
||||||
|
|
||||||
|
# For Cygwin, switch paths to Windows format before running java |
||||||
|
if $cygwin ; then |
||||||
|
APP_HOME=`cygpath --path --mixed "$APP_HOME"` |
||||||
|
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` |
||||||
|
JAVACMD=`cygpath --unix "$JAVACMD"` |
||||||
|
|
||||||
|
# We build the pattern for arguments to be converted via cygpath |
||||||
|
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` |
||||||
|
SEP="" |
||||||
|
for dir in $ROOTDIRSRAW ; do |
||||||
|
ROOTDIRS="$ROOTDIRS$SEP$dir" |
||||||
|
SEP="|" |
||||||
|
done |
||||||
|
OURCYGPATTERN="(^($ROOTDIRS))" |
||||||
|
# Add a user-defined pattern to the cygpath arguments |
||||||
|
if [ "$GRADLE_CYGPATTERN" != "" ] ; then |
||||||
|
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" |
||||||
|
fi |
||||||
|
# Now convert the arguments - kludge to limit ourselves to /bin/sh |
||||||
|
i=0 |
||||||
|
for arg in "$@" ; do |
||||||
|
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` |
||||||
|
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option |
||||||
|
|
||||||
|
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition |
||||||
|
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` |
||||||
|
else |
||||||
|
eval `echo args$i`="\"$arg\"" |
||||||
|
fi |
||||||
|
i=$((i+1)) |
||||||
|
done |
||||||
|
case $i in |
||||||
|
(0) set -- ;; |
||||||
|
(1) set -- "$args0" ;; |
||||||
|
(2) set -- "$args0" "$args1" ;; |
||||||
|
(3) set -- "$args0" "$args1" "$args2" ;; |
||||||
|
(4) set -- "$args0" "$args1" "$args2" "$args3" ;; |
||||||
|
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; |
||||||
|
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; |
||||||
|
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; |
||||||
|
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; |
||||||
|
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; |
||||||
|
esac |
||||||
|
fi |
||||||
|
|
||||||
|
# Escape application args |
||||||
|
save () { |
||||||
|
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done |
||||||
|
echo " " |
||||||
|
} |
||||||
|
APP_ARGS=$(save "$@") |
||||||
|
|
||||||
|
# Collect all arguments for the java command, following the shell quoting and substitution rules |
||||||
|
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" |
||||||
|
|
||||||
|
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong |
||||||
|
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then |
||||||
|
cd "$(dirname "$0")" |
||||||
|
fi |
||||||
|
|
||||||
|
exec "$JAVACMD" "$@" |
@ -0,0 +1,84 @@ |
|||||||
|
@if "%DEBUG%" == "" @echo off |
||||||
|
@rem ########################################################################## |
||||||
|
@rem |
||||||
|
@rem Gradle startup script for Windows |
||||||
|
@rem |
||||||
|
@rem ########################################################################## |
||||||
|
|
||||||
|
@rem Set local scope for the variables with windows NT shell |
||||||
|
if "%OS%"=="Windows_NT" setlocal |
||||||
|
|
||||||
|
set DIRNAME=%~dp0 |
||||||
|
if "%DIRNAME%" == "" set DIRNAME=. |
||||||
|
set APP_BASE_NAME=%~n0 |
||||||
|
set APP_HOME=%DIRNAME% |
||||||
|
|
||||||
|
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. |
||||||
|
set DEFAULT_JVM_OPTS= |
||||||
|
|
||||||
|
@rem Find java.exe |
||||||
|
if defined JAVA_HOME goto findJavaFromJavaHome |
||||||
|
|
||||||
|
set JAVA_EXE=java.exe |
||||||
|
%JAVA_EXE% -version >NUL 2>&1 |
||||||
|
if "%ERRORLEVEL%" == "0" goto init |
||||||
|
|
||||||
|
echo. |
||||||
|
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. |
||||||
|
echo. |
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the |
||||||
|
echo location of your Java installation. |
||||||
|
|
||||||
|
goto fail |
||||||
|
|
||||||
|
:findJavaFromJavaHome |
||||||
|
set JAVA_HOME=%JAVA_HOME:"=% |
||||||
|
set JAVA_EXE=%JAVA_HOME%/bin/java.exe |
||||||
|
|
||||||
|
if exist "%JAVA_EXE%" goto init |
||||||
|
|
||||||
|
echo. |
||||||
|
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% |
||||||
|
echo. |
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the |
||||||
|
echo location of your Java installation. |
||||||
|
|
||||||
|
goto fail |
||||||
|
|
||||||
|
:init |
||||||
|
@rem Get command-line arguments, handling Windows variants |
||||||
|
|
||||||
|
if not "%OS%" == "Windows_NT" goto win9xME_args |
||||||
|
|
||||||
|
:win9xME_args |
||||||
|
@rem Slurp the command line arguments. |
||||||
|
set CMD_LINE_ARGS= |
||||||
|
set _SKIP=2 |
||||||
|
|
||||||
|
:win9xME_args_slurp |
||||||
|
if "x%~1" == "x" goto execute |
||||||
|
|
||||||
|
set CMD_LINE_ARGS=%* |
||||||
|
|
||||||
|
:execute |
||||||
|
@rem Setup the command line |
||||||
|
|
||||||
|
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar |
||||||
|
|
||||||
|
@rem Execute Gradle |
||||||
|
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% |
||||||
|
|
||||||
|
:end |
||||||
|
@rem End local scope for the variables with windows NT shell |
||||||
|
if "%ERRORLEVEL%"=="0" goto mainEnd |
||||||
|
|
||||||
|
:fail |
||||||
|
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of |
||||||
|
rem the _cmd.exe /c_ return code! |
||||||
|
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 |
||||||
|
exit /b 1 |
||||||
|
|
||||||
|
:mainEnd |
||||||
|
if "%OS%"=="Windows_NT" endlocal |
||||||
|
|
||||||
|
:omega |
@ -0,0 +1 @@ |
|||||||
|
include ':app', ':app-updater', ':app-updater', ':app-dialog' |