├── .gitignore
├── LICENSE.txt
├── README.md
├── build.gradle
├── demo
├── .gitignore
├── build.gradle
├── proguard-rules.pro
└── src
│ ├── androidTest
│ └── java
│ │ └── com
│ │ └── netease
│ │ └── heartouch
│ │ └── example
│ │ └── ApplicationTest.java
│ ├── main
│ ├── AndroidManifest.xml
│ ├── java
│ │ └── com
│ │ │ └── netease
│ │ │ └── hearttouch
│ │ │ └── example
│ │ │ ├── HTApplication.java
│ │ │ ├── MainActivity.java
│ │ │ ├── PayActivity.java
│ │ │ ├── ProductDetailActivity.java
│ │ │ └── WebActivity.java
│ └── res
│ │ ├── anim
│ │ ├── enter.xml
│ │ ├── enter1.xml
│ │ └── exit.xml
│ │ ├── layout
│ │ ├── activity_pay.xml
│ │ ├── activity_product_detail.xml
│ │ ├── activity_web.xml
│ │ ├── content_pay.xml
│ │ ├── content_product_detail.xml
│ │ └── content_web.xml
│ │ ├── menu
│ │ └── menu_main.xml
│ │ ├── mipmap-hdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-mdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xhdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xxhdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xxxhdpi
│ │ └── ic_launcher.png
│ │ ├── values-v21
│ │ └── styles.xml
│ │ ├── values-w820dp
│ │ └── dimens.xml
│ │ └── values
│ │ ├── colors.xml
│ │ ├── dimens.xml
│ │ ├── strings.xml
│ │ └── styles.xml
│ └── test
│ └── java
│ └── com
│ └── netease
│ └── heartouch
│ └── example
│ └── ExampleUnitTest.java
├── gradle.properties
├── gradle
└── wrapper
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── gradlew
├── gradlew.bat
├── htrouterdispatch-process
├── .gitignore
├── build.gradle
└── src
│ └── main
│ └── java
│ └── com
│ └── netease
│ └── hearttouch
│ └── router
│ ├── HTAnnotatedClass.java
│ ├── HTClassValidator.java
│ ├── HTCodeGenerator.java
│ └── HTRouterDispatchProcess.java
├── htrouterdispatch
├── .gitignore
├── build.gradle
└── src
│ └── main
│ └── java
│ └── com
│ └── netease
│ └── hearttouch
│ └── router
│ ├── HTRouter.java
│ └── HTRouterEntry.java
└── settings.gradle
/.gitignore:
--------------------------------------------------------------------------------
1 |
2 | local.properties
3 |
4 | .idea/
5 |
6 | .gradle/
7 | *.iml
8 |
--------------------------------------------------------------------------------
/LICENSE.txt:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2016 Netease Inc.
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # HT Universal Router页面管理框架
2 | `HT Universal Router`是一种基于注解的Android页面管理框架,通过URL进行页面跳转管理,同时可以设置进入及退出动画,并提供页面降级及跳转拦截和重定向功能。
3 | ## 基本功能
4 | - 通过`URL`进行`Activity`之间跳转
5 | - 参数传递 (复杂类型参数传递)
6 | - 进入退出动画资源自定义
7 | - 页面降级 (注: 降级指的是`Native`无法匹配的`URL`可以转为通过`Webview`处理)
8 | - `Webview`跳转`Native`识别
9 | - `URL`中参数自动转化`Intent`参数
10 | - 支持`startActivity`和`startActivityForResult`
11 | - `URL`与`Activity`多对一映射
12 | - 支持`Debug`模式,日志输出
13 | - 支持适配Android M DeepLink
14 | - 支持全局跳转数据,处理每次跳转监听 用户打点统计等功能
15 | - 支持自定义`WebActivity`接收`URL`的`KEY`
16 |
17 | ## 用法
18 |
19 | ### AndroidManifest.xml
20 | 添加权限:
21 |
22 | ```
23 |
24 | ```
25 | 注册`HTRouterActivity`,根据实际修改`data`标签中的信息
26 |
27 | ```
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 | ```
41 | 相应的降级`WebActivity`,定义示例见后文。
42 |
43 | ```
44 |
48 | ```
49 |
50 | ### 初始化
51 | 在`Application`的`onCreate()`中添加如下信息,其中`HTRouterActivity`、`HTLogUtil`、`HTRouterManager`、`HTRouterHandler`、`HTRouterHandlerParams`编译后会自动生成(注:仅当有页面使用`@HTRouter`注解时,以上代码才会生成)。
52 |
53 | - 降级页面接收的`URL`的`key`
54 | - 注册降级页面相应的`WebviewActivity`;
55 | - 是否开启`debug`模式
56 | - 每次跳转的监听,可以进行自定义处理,包括跳转的拦截和重定义等,可以作为拦截器使用,通过返回true来阻止本管理框架进行跳转,通过修改`HTRouterHandlerParams`中的内容来修改跳转的目标。
57 |
58 | ```
59 | HTRouterManager.init();
60 | //注册绑定默认的降级页面
61 | StringcustomUrlKey = "customUrlKey"; //HTWebActivity接收参数key
62 | HTRouterManager.registerWebActivity(WebActivity.class, customUrlKey);
63 | //开启Debug模式,输出相应日志
64 | HTRouterManager.setDebugMode(true);
65 | HTRouterManager.setHtRouterHandler(new HTRouterHandler() {
66 | @Override
67 | public boolean handleRoute(Context context, HTRouterHandlerParams routerParams) {
68 | HTLogUtil.d("统计数据:" + context.getClass().getSimpleName() + "-->跳转url-->" + routerParams.url + " 参数intent" + routerParams.sourceIntent);
69 | //如果需要拦截或者改变跳转的目标可以直接改变url或者sourceIntent
70 | routerParams.url = "http://www.kaola.com/pay";
71 | //如果返回true则表示由监听中进行处理,不需要HTRouter负责跳转
72 | return false;
73 | }
74 | });
75 |
76 | ```
77 |
78 | ### 注解使用
79 | `@HTRouter` 字段支持`Activity`类注解:
80 |
81 | - `url`: 支持url数组 {"http://kaola.com","http://m.kaola.com"}
82 | - `entryAnim` :`activity`进入动画 `R.anim.entry`
83 | - `exitAnim` :`activity`退出动画 `R.anim.exit`
84 |
85 | 例如:
86 |
87 | ```
88 | @HTRouter(url = {"http://www.kaola.com/pay"},entryAnim = R.anim.enter,exitAnim = R.anim.exit)
89 | ```
90 |
91 | ### 页面跳转方法
92 | 1. 使用`HTRouterManager.startActivity()`接口
93 |
94 | - `activity`: 当前页面的`activity`
95 | - `url`: 跳转的目标URL
96 | - `sourceIntent`: 传递进来一个intent,用于传递用户数据及启动模式等扩展
97 | - `isFinish`: 跳转后是否需要关闭当前页面
98 |
99 | 例如`HTRouterManager.startActivity(this, "http://www.kaola.com/pay", null, false);`
100 |
101 | 2. 还提供`HTRouterManager.startActivityForResult()`,参数与`HTRouterManager.startActivity()`相同,回调发生在`Activity`中。
102 | 3. 由于`Fragment`的`startActivityForResult()`的回调可以选择发生在`Fragment`或者`Fragment`所处的`Activity`中(通常直接调用`startActivityForResult()`发生在`fragment`中,而通过`getActivity().startActivityForResult()`则发生在Activity中),故另外提供`HTRouterManager.startActivityForResult(Fragment fragment, String url, Intent sourceIntent, boolean isFinish)`接口,该方法只有第一个参数不同。
103 |
104 | ### 参数获取
105 | 除了在页面跳转的`sourceIntent`中传入的自定义类型参数外,本框架还实现`URL`中参数字段的自动解析,解析后保存在`HashMap`中序列化后放在跳转的`Intent`参数中。用户可以用如下方式获取,需要进行强制类型转换:
106 |
107 | ```
108 | Intent intent = getIntent();
109 | HashMap urlParamsMap = (HashMap)intent.getSerializableExtra(HTRouterManager.HT_URL_PARAMS_KEY);
110 | ```
111 |
112 | ### 降级及升级WebActivity示例
113 | * 降级`WebActivity`,`Native`处理不了的转`Webview`,只需要接收`url`参数后`loadUrl`一下。
114 |
115 | ```
116 | public class WebActivity extends AppCompatActivity {
117 | @Override
118 | protected void onCreate(Bundle savedInstanceState) {
119 | super.onCreate(savedInstanceState);
120 | Intent intent = getIntent();
121 | String url = intent.getStringExtra("customUrlKey");
122 | WebView webView = new WebView(this);
123 | webView.getSettings().setJavaScriptEnabled(true);
124 | webView.getSettings().setUseWideViewPort(true);
125 | webView.getSettings().setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN);
126 | webView.getSettings().setCacheMode(WebSettings.LOAD_NO_CACHE);
127 | webView.getSettings().setLoadWithOverviewMode(true);
128 | String targetUrl = HTRouterManager.getHttpSchemaHostAndPath(url);
129 | HTLogUtil.d("打开链接=" + targetUrl);
130 | webView.loadUrl(targetUrl);
131 | setContentView(webView);
132 | }
133 | }
134 | ```
135 |
136 | * 升级`WebActivity`,`webview`中的`url`先交给`native`处理,需要重载`shouldOverrideUrlLoading()`,处理`webview`中的`url`。
137 |
138 | ```
139 | public class WebActivity extends AppCompatActivity {
140 | @Override
141 | protected void onCreate(Bundle savedInstanceState) {
142 | super.onCreate(savedInstanceState);
143 | //...
144 | //一些初始化...
145 | webView.setWebViewClient(new WebViewClient() {
146 | @Override
147 | public boolean shouldOverrideUrlLoading(WebView view, String url) {
148 | Log.i("HT", "重新加载=" + url);
149 | HTRouterEntry entity = HTRouterManager.findRouterEntryByUrl(url);
150 | //为了防止匹配不上后循环跳,这里需要有个判断
151 | if (entity != null) {
152 | //匹配上之后
153 | HTRouterManager.startActivity(WebActivity.this, url, null, false);
154 | return true;
155 | }
156 | return super.shouldOverrideUrlLoading(view, url);
157 | }
158 | });
159 | //...
160 | //一些后续操作
161 | }
162 | }
163 | ```
164 |
165 | ### 进阶使用
166 | 本框架可以注册`HTRouterHandler`,进行跳转拦截或记录等处理,实现该接口并通过`HTRouterManager.setHtRouterHandler()`注册。返回值`true`代表拦截处理,则本框架不会进行后续的处理。返回`false`代表还需要本框架进行进一步跳转处理,同时可以修改`routerParams`中的`url`和`sourceIntent`来达到动态控制跳转的重定向。
167 |
168 | ```
169 | public interface HTRouterHandler {
170 | boolean handleRoute(Context context, HTRouterHandlerParams routerParams);
171 | }
172 | ```
173 | ## 集成
174 | ### Gradle
175 | `app`的`build.grade`中添加
176 |
177 | ```
178 | apply plugin: 'com.neenbedankt.android-apt'
179 |
180 | dependencies {
181 | compile 'com.netease.hearttouch:ht-universalrouter-dispatch:0.1.0'
182 | apt 'com.netease.hearttouch:ht-universalrouter-dispatch-process:0.1.0'
183 | }
184 | ```
185 |
186 | `project`的`build.gradle`的`dependencies`下面添加
187 |
188 | ```
189 | classpath 'com.neenbedankt.gradle.plugins:android-apt:1.4'
190 | ```
191 |
192 | ### maven
193 | ```
194 |
195 |
196 | com.netease.hearttouch
197 | ht-universalrouter-dispatch
198 | 0.1.0
199 |
200 |
201 | com.netease.hearttouch
202 | ht-universalrouter-dispatch-process
203 | 0.1.0
204 | true
205 |
206 |
207 | ```
208 | ## 代码混淆
209 | 如果要使用混淆,在引用工程的`proguard`文件中,添加如下代码:
210 |
211 | ```
212 | -keep class com.netease.hearttouch.router.** { *; }
213 | ```
214 |
215 | ## 许可证
216 | `HT Universal Router` 使用 `MIT` 许可证,详情见 [LICENSE](https://github.com/NEYouFan/ht-universalrouter-android/blob/master/LICENSE.txt) 文件。
--------------------------------------------------------------------------------
/build.gradle:
--------------------------------------------------------------------------------
1 | // Top-level build file where you can add configuration options common to all sub-projects/modules.
2 |
3 | buildscript {
4 | repositories {
5 | jcenter()
6 | }
7 | dependencies {
8 | classpath 'com.android.tools.build:gradle:1.5.0'
9 | classpath 'com.neenbedankt.gradle.plugins:android-apt:1.4'
10 | // NOTE: Do not place your application dependencies here; they belong
11 | // in the individual module build.gradle files
12 | }
13 | }
14 |
15 | allprojects {
16 | repositories {
17 | jcenter()
18 | }
19 | }
20 |
21 | task clean(type: Delete) {
22 | delete rootProject.buildDir
23 | }
24 |
--------------------------------------------------------------------------------
/demo/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/demo/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 | apply plugin: 'com.neenbedankt.android-apt'
3 |
4 | android {
5 | compileSdkVersion 23
6 | buildToolsVersion "23.0.2"
7 |
8 | defaultConfig {
9 | applicationId "com.netease.heartouch.example"
10 | minSdkVersion 14
11 | targetSdkVersion 23
12 | versionCode 1
13 | versionName "1.0"
14 | }
15 | buildTypes {
16 | debug {
17 | minifyEnabled true
18 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
19 | }
20 |
21 | release {
22 | minifyEnabled true
23 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
24 | }
25 | }
26 | }
27 |
28 | dependencies {
29 | compile fileTree(dir: 'libs', include: ['*.jar'])
30 | testCompile 'junit:junit:4.12'
31 | compile 'com.android.support:appcompat-v7:23.1.1'
32 | compile 'com.android.support:design:23.1.1'
33 | }
34 |
35 | dependencies {
36 | compile project(':htrouterdispatch')
37 | apt project(':htrouterdispatch-process')
38 | }
--------------------------------------------------------------------------------
/demo/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in D:\UserSpace\Android\AndroidSDK/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 | -keep class com.netease.heartouch.router.** { *; }
19 | -dontwarn okio.**
--------------------------------------------------------------------------------
/demo/src/androidTest/java/com/netease/heartouch/example/ApplicationTest.java:
--------------------------------------------------------------------------------
1 | package com.netease.heartouch.example;
2 |
3 | import android.app.Application;
4 | import android.test.ApplicationTestCase;
5 |
6 | /**
7 | * Testing Fundamentals
8 | */
9 | public class ApplicationTest extends ApplicationTestCase {
10 | public ApplicationTest() {
11 | super(Application.class);
12 | }
13 | }
--------------------------------------------------------------------------------
/demo/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
12 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
40 |
44 |
48 |
49 |
50 |
51 |
--------------------------------------------------------------------------------
/demo/src/main/java/com/netease/hearttouch/example/HTApplication.java:
--------------------------------------------------------------------------------
1 | package com.netease.hearttouch.example;
2 |
3 | import android.app.Application;
4 | import android.content.Context;
5 |
6 | import com.netease.hearttouch.router.HTLogUtil;
7 | import com.netease.hearttouch.router.HTRouterHandler;
8 | import com.netease.hearttouch.router.HTRouterHandlerParams;
9 | import com.netease.hearttouch.router.HTRouterManager;
10 |
11 | /**
12 | * @author hzshengxueming
13 | */
14 |
15 | public class HTApplication extends Application {
16 | @Override
17 | public void onCreate() {
18 | super.onCreate();
19 | HTRouterManager.init();
20 | //注册绑定默认的降级页面
21 | String customUrlKey = "customUrlKey"; //HTWebActivity接收参数key
22 | HTRouterManager.registerWebActivity(WebActivity.class, customUrlKey);
23 | //开启Debug模式,输出相应日志
24 | HTRouterManager.setDebugMode(true);
25 | //处理每次跳转监听 用户打点统计等
26 | HTRouterManager.setHtRouterHandler(new HTRouterHandler() {
27 | @Override
28 | public boolean handleRoute(Context context, HTRouterHandlerParams routerParams) {
29 | HTLogUtil.d("统计数据:" + context.getClass().getSimpleName() + "-->跳转url-->" + routerParams.url + " 参数intent" + routerParams.sourceIntent);
30 | //如果需要拦截或者改变跳转的目标可以直接改变url或者sourceIntent
31 | // routerParams.url = "http://www.kaola.com/pay?a=b&c=d";
32 | //如果返回true则表示由监听中进行处理,不需要HTRouter负责跳转
33 | return false;
34 | }
35 | });
36 | }
37 | }
38 |
--------------------------------------------------------------------------------
/demo/src/main/java/com/netease/hearttouch/example/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.netease.hearttouch.example;
2 |
3 | import android.os.Bundle;
4 | import android.support.v7.app.AppCompatActivity;
5 | import android.view.View;
6 | import android.widget.Button;
7 |
8 | import com.netease.hearttouch.router.HTRouter;
9 | import com.netease.hearttouch.router.HTRouterManager;
10 |
11 | @HTRouter(url = {"kaola://www.kaola.com/","kaola://m.kaola.com"},entryAnim = R.anim.enter,exitAnim = R.anim.exit)
12 | public class MainActivity extends AppCompatActivity {
13 |
14 | @Override
15 | protected void onCreate(Bundle savedInstanceState) {
16 | super.onCreate(savedInstanceState);
17 | Button button = new Button(this);
18 | button.setText("跳转到考拉商品类别H5页面");
19 | button.setOnClickListener(new View.OnClickListener() {
20 | @Override
21 | public void onClick(View v) {
22 | HTRouterManager.startActivity(MainActivity.this, "http://www.kaola.com/activity/detail/6932.shtml?navindex=7", null, true);
23 | }
24 | });
25 | setContentView(button);
26 | }
27 |
28 | }
29 |
--------------------------------------------------------------------------------
/demo/src/main/java/com/netease/hearttouch/example/PayActivity.java:
--------------------------------------------------------------------------------
1 | package com.netease.hearttouch.example;
2 |
3 | import android.os.Bundle;
4 | import android.support.v7.app.AppCompatActivity;
5 | import android.view.View;
6 | import android.widget.Button;
7 |
8 | import com.netease.hearttouch.router.HTRouter;
9 |
10 | @HTRouter(url = {"http://www.kaola.com/pay"}, entryAnim = R.anim.enter, exitAnim = R.anim.exit)
11 | public class PayActivity extends AppCompatActivity {
12 |
13 | @Override
14 | protected void onCreate(Bundle savedInstanceState) {
15 | super.onCreate(savedInstanceState);
16 | setContentView(R.layout.activity_pay);
17 | Button button = (Button) findViewById(R.id.id_close_pay_page);
18 | button.setOnClickListener(new View.OnClickListener() {
19 | @Override
20 | public void onClick(View v) {
21 | finish();
22 | }
23 | });
24 | }
25 |
26 | }
27 |
--------------------------------------------------------------------------------
/demo/src/main/java/com/netease/hearttouch/example/ProductDetailActivity.java:
--------------------------------------------------------------------------------
1 |
2 | package com.netease.hearttouch.example;
3 |
4 | import android.content.Intent;
5 | import android.os.Bundle;
6 | import android.support.v7.app.AppCompatActivity;
7 | import android.util.Log;
8 | import android.view.View;
9 | import android.webkit.WebSettings;
10 | import android.webkit.WebView;
11 | import android.webkit.WebViewClient;
12 | import android.widget.Button;
13 | import android.widget.TextView;
14 | import android.widget.Toast;
15 |
16 | import com.netease.hearttouch.router.HTRouter;
17 | import com.netease.hearttouch.router.HTRouterEntry;
18 | import com.netease.hearttouch.router.HTRouterManager;
19 |
20 | import java.util.HashMap;
21 |
22 | @HTRouter(url = {"http://www.kaola.com/activity/detail/{id}.shtml","http://m.kaola.com/activity/detail/{id}.shtml","http://m.kaola.com/product/{id}.html","http://www.kaola.com/product/{id}.html"},entryAnim = R.anim.enter,exitAnim = R.anim.exit)
23 | public class ProductDetailActivity extends AppCompatActivity implements View.OnClickListener{
24 | private TextView tv;
25 | private Button btn;
26 | private Button btn1;
27 | @Override
28 | protected void onCreate(Bundle savedInstanceState) {
29 | super.onCreate(savedInstanceState);
30 | setContentView(R.layout.activity_product_detail);
31 |
32 |
33 | tv = (TextView) findViewById(R.id.tv);
34 | btn= (Button) findViewById(R.id.btn);
35 | btn.setOnClickListener(this);
36 | btn1 = (Button) findViewById(R.id.btn1);
37 | btn1.setOnClickListener(this);
38 | WebView webView = (WebView) findViewById(R.id.webview);
39 |
40 | //获取URL参数
41 | Intent intent = getIntent();
42 | HashMap urlParamsMap = (HashMap)intent.getSerializableExtra(HTRouterManager.HT_URL_PARAMS_KEY);
43 | for(String key : urlParamsMap.keySet()){
44 | Log.d("url_params", key + ":" + urlParamsMap.get(key));
45 | }
46 |
47 | webView.getSettings().setJavaScriptEnabled(true);
48 | webView.getSettings().setUseWideViewPort(true);
49 | webView.getSettings().setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN);
50 | webView.getSettings().setCacheMode(WebSettings.LOAD_NO_CACHE);
51 | webView.getSettings().setUseWideViewPort(true);
52 | webView.getSettings().setLoadWithOverviewMode(true);
53 | webView.setWebViewClient(new WebViewClient() {
54 | @Override
55 | public boolean shouldOverrideUrlLoading(WebView view, String url) {
56 | Log.i("HT", "重新加载=" + url);
57 | HTRouterEntry entity = HTRouterManager.findRouterEntryByUrl(url);
58 | //为了防止匹配不上后循环跳,这里需要有个判断
59 | if (entity != null) {
60 | HTRouterManager.startActivity(ProductDetailActivity.this, url, null, false);
61 | return true;
62 | }
63 | return super.shouldOverrideUrlLoading(view, url);
64 | }
65 | });
66 | if (intent!=null){
67 | Log.i("HT", "产品详情=" + intent.getData());
68 | tv.setText("产品详情=" + intent.getData());
69 | HTRouterEntry entity = HTRouterManager.findRouterEntryByUrl(intent.getData().toString());
70 | webView.loadUrl(intent.getData().toString());
71 | }
72 |
73 |
74 | }
75 |
76 | @Override
77 | public void onClick(View v) {
78 | switch (v.getId()){
79 | case R.id.btn:
80 | Intent sourceIntent = new Intent();
81 | sourceIntent.putExtra("price",12345);
82 | sourceIntent.putExtra("id","m123456");
83 | sourceIntent.putExtra("name","Mac Pro 13寸");
84 | HTRouterManager.startActivityForResult(ProductDetailActivity.this,"http://www.kaola.com/pay",sourceIntent,false,1001);
85 | break;
86 | case R.id.btn1:
87 | Intent sourceIntent1 = new Intent();
88 | sourceIntent1.putExtra("price",12345);
89 | sourceIntent1.putExtra("id","m123456");
90 | sourceIntent1.putExtra("name","Mac Pro 13寸");
91 | HTRouterManager.startActivityForResult(ProductDetailActivity.this,
92 | "http://www.kaola.com/mall/paygame.html",
93 | sourceIntent1, false, 1001);
94 | break;
95 | }
96 | }
97 |
98 | @Override
99 | protected void onActivityResult(int requestCode, int resultCode, Intent data) {
100 | switch (requestCode){
101 | case 1001:
102 | Toast.makeText(ProductDetailActivity.this,"支付返回",Toast.LENGTH_SHORT).show();
103 | break;
104 | }
105 | }
106 | }
107 |
--------------------------------------------------------------------------------
/demo/src/main/java/com/netease/hearttouch/example/WebActivity.java:
--------------------------------------------------------------------------------
1 |
2 | package com.netease.hearttouch.example;
3 |
4 | import android.content.Intent;
5 | import android.os.Bundle;
6 | import android.support.v7.app.AppCompatActivity;
7 | import android.util.Log;
8 | import android.webkit.WebSettings;
9 | import android.webkit.WebView;
10 | import android.webkit.WebViewClient;
11 |
12 | import com.netease.hearttouch.router.HTLogUtil;
13 | import com.netease.hearttouch.router.HTRouterEntry;
14 | import com.netease.hearttouch.router.HTRouterManager;
15 |
16 | public class WebActivity extends AppCompatActivity {
17 |
18 | @Override
19 | protected void onCreate(Bundle savedInstanceState) {
20 | super.onCreate(savedInstanceState);
21 | Intent intent = getIntent();
22 | String url = intent.getStringExtra("customUrlKey");
23 | WebView webView = new WebView(this);
24 | webView.getSettings().setJavaScriptEnabled(true);
25 | webView.getSettings().setUseWideViewPort(true);
26 | webView.getSettings().setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN);
27 | webView.getSettings().setCacheMode(WebSettings.LOAD_NO_CACHE);
28 | webView.getSettings().setUseWideViewPort(true);
29 | webView.getSettings().setLoadWithOverviewMode(true);
30 | webView.setWebViewClient(new WebViewClient() {
31 | @Override
32 | public boolean shouldOverrideUrlLoading(WebView view, String url) {
33 | Log.i("HT", "重新加载=" + url);
34 | HTRouterEntry entity = HTRouterManager.findRouterEntryByUrl(url);
35 | //为了防止匹配不上后循环跳,这里需要有个判断
36 | if (entity != null) {
37 | HTRouterManager.startActivity(WebActivity.this, url, null, false);
38 | return true;
39 | }
40 | return super.shouldOverrideUrlLoading(view, url);
41 | }
42 | });
43 | String targetUrl = HTRouterManager.getHttpSchemaHostAndPath(url);
44 | HTLogUtil.d("打开链接=" + targetUrl);
45 | webView.loadUrl(targetUrl);
46 | setContentView(webView);
47 | }
48 | }
49 |
--------------------------------------------------------------------------------
/demo/src/main/res/anim/enter.xml:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/demo/src/main/res/anim/enter1.xml:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/demo/src/main/res/anim/exit.xml:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/demo/src/main/res/layout/activity_pay.xml:
--------------------------------------------------------------------------------
1 |
2 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
--------------------------------------------------------------------------------
/demo/src/main/res/layout/activity_product_detail.xml:
--------------------------------------------------------------------------------
1 |
2 |
9 |
10 |
11 |
12 |
13 |
--------------------------------------------------------------------------------
/demo/src/main/res/layout/activity_web.xml:
--------------------------------------------------------------------------------
1 |
2 |
9 |
10 |
14 |
15 |
21 |
22 |
23 |
24 |
25 |
26 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/demo/src/main/res/layout/content_pay.xml:
--------------------------------------------------------------------------------
1 |
2 |
15 |
19 |
24 |
25 |
--------------------------------------------------------------------------------
/demo/src/main/res/layout/content_product_detail.xml:
--------------------------------------------------------------------------------
1 |
2 |
15 |
19 |
24 |
29 |
33 |
34 |
--------------------------------------------------------------------------------
/demo/src/main/res/layout/content_web.xml:
--------------------------------------------------------------------------------
1 |
2 |
14 |
15 |
16 |
--------------------------------------------------------------------------------
/demo/src/main/res/menu/menu_main.xml:
--------------------------------------------------------------------------------
1 |
11 |
--------------------------------------------------------------------------------
/demo/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NEYouFan/ht-universalrouter-android/f6ae0fe3b7c5431c1199ed1140fb0fc1ce3a860a/demo/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/demo/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NEYouFan/ht-universalrouter-android/f6ae0fe3b7c5431c1199ed1140fb0fc1ce3a860a/demo/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/demo/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NEYouFan/ht-universalrouter-android/f6ae0fe3b7c5431c1199ed1140fb0fc1ce3a860a/demo/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/demo/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NEYouFan/ht-universalrouter-android/f6ae0fe3b7c5431c1199ed1140fb0fc1ce3a860a/demo/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/demo/src/main/res/mipmap-xxxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NEYouFan/ht-universalrouter-android/f6ae0fe3b7c5431c1199ed1140fb0fc1ce3a860a/demo/src/main/res/mipmap-xxxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/demo/src/main/res/values-v21/styles.xml:
--------------------------------------------------------------------------------
1 | >
2 |
3 |
9 |
10 |
--------------------------------------------------------------------------------
/demo/src/main/res/values-w820dp/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 | 64dp
6 |
7 |
--------------------------------------------------------------------------------
/demo/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #3F51B5
4 | #303F9F
5 | #FF4081
6 |
7 |
--------------------------------------------------------------------------------
/demo/src/main/res/values/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 16dp
4 | 16dp
5 | 16dp
6 |
7 |
--------------------------------------------------------------------------------
/demo/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | HTRouter
3 | Settings
4 | ProductDetailActivity
5 | PayActivity
6 | WebActivity
7 |
8 |
--------------------------------------------------------------------------------
/demo/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
10 |
11 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
--------------------------------------------------------------------------------
/demo/src/test/java/com/netease/heartouch/example/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.netease.heartouch.example;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * To work on unit tests, switch the Test Artifact in the Build Variants view.
9 | */
10 | public class ExampleUnitTest {
11 | @Test
12 | public void addition_isCorrect() throws Exception {
13 | assertEquals(4, 2 + 2);
14 | }
15 | }
--------------------------------------------------------------------------------
/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/NEYouFan/ht-universalrouter-android/f6ae0fe3b7c5431c1199ed1140fb0fc1ce3a860a/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Wed Oct 21 11:34:03 PDT 2015
2 | distributionBase=GRADLE_USER_HOME
3 | distributionPath=wrapper/dists
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.4-all.zip
7 |
--------------------------------------------------------------------------------
/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # Attempt to set APP_HOME
46 | # Resolve links: $0 may be a link
47 | PRG="$0"
48 | # Need this for relative symlinks.
49 | while [ -h "$PRG" ] ; do
50 | ls=`ls -ld "$PRG"`
51 | link=`expr "$ls" : '.*-> \(.*\)$'`
52 | if expr "$link" : '/.*' > /dev/null; then
53 | PRG="$link"
54 | else
55 | PRG=`dirname "$PRG"`"/$link"
56 | fi
57 | done
58 | SAVED="`pwd`"
59 | cd "`dirname \"$PRG\"`/" >/dev/null
60 | APP_HOME="`pwd -P`"
61 | cd "$SAVED" >/dev/null
62 |
63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
64 |
65 | # Determine the Java command to use to start the JVM.
66 | if [ -n "$JAVA_HOME" ] ; then
67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
68 | # IBM's JDK on AIX uses strange locations for the executables
69 | JAVACMD="$JAVA_HOME/jre/sh/java"
70 | else
71 | JAVACMD="$JAVA_HOME/bin/java"
72 | fi
73 | if [ ! -x "$JAVACMD" ] ; then
74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
75 |
76 | Please set the JAVA_HOME variable in your environment to match the
77 | location of your Java installation."
78 | fi
79 | else
80 | JAVACMD="java"
81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
82 |
83 | Please set the JAVA_HOME variable in your environment to match the
84 | location of your Java installation."
85 | fi
86 |
87 | # Increase the maximum file descriptors if we can.
88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
89 | MAX_FD_LIMIT=`ulimit -H -n`
90 | if [ $? -eq 0 ] ; then
91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
92 | MAX_FD="$MAX_FD_LIMIT"
93 | fi
94 | ulimit -n $MAX_FD
95 | if [ $? -ne 0 ] ; then
96 | warn "Could not set maximum file descriptor limit: $MAX_FD"
97 | fi
98 | else
99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
100 | fi
101 | fi
102 |
103 | # For Darwin, add options to specify how the application appears in the dock
104 | if $darwin; then
105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
106 | fi
107 |
108 | # For Cygwin, switch paths to Windows format before running java
109 | if $cygwin ; then
110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
112 | JAVACMD=`cygpath --unix "$JAVACMD"`
113 |
114 | # We build the pattern for arguments to be converted via cygpath
115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
116 | SEP=""
117 | for dir in $ROOTDIRSRAW ; do
118 | ROOTDIRS="$ROOTDIRS$SEP$dir"
119 | SEP="|"
120 | done
121 | OURCYGPATTERN="(^($ROOTDIRS))"
122 | # Add a user-defined pattern to the cygpath arguments
123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
125 | fi
126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
127 | i=0
128 | for arg in "$@" ; do
129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
131 |
132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
134 | else
135 | eval `echo args$i`="\"$arg\""
136 | fi
137 | i=$((i+1))
138 | done
139 | case $i in
140 | (0) set -- ;;
141 | (1) set -- "$args0" ;;
142 | (2) set -- "$args0" "$args1" ;;
143 | (3) set -- "$args0" "$args1" "$args2" ;;
144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
150 | esac
151 | fi
152 |
153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
154 | function splitJvmOpts() {
155 | JVM_OPTS=("$@")
156 | }
157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
159 |
160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
161 |
--------------------------------------------------------------------------------
/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/htrouterdispatch-process/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/htrouterdispatch-process/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'java'
2 |
3 | sourceCompatibility = JavaVersion.VERSION_1_7
4 | targetCompatibility = JavaVersion.VERSION_1_7
5 |
6 | dependencies {
7 | compile project (':htrouterdispatch')
8 | compile 'com.google.auto.service:auto-service:1.0-rc2'
9 | compile 'com.squareup:javapoet:1.0.0'
10 | }
--------------------------------------------------------------------------------
/htrouterdispatch-process/src/main/java/com/netease/hearttouch/router/HTAnnotatedClass.java:
--------------------------------------------------------------------------------
1 | /*
2 | * This source code is licensed under the MIT-style license found in the
3 | * LICENSE file in the root directory of this source tree.
4 | */
5 | package com.netease.hearttouch.router;
6 |
7 | import javax.lang.model.element.TypeElement;
8 |
9 | /**
10 | * 用于记录被标注的类的设置
11 | * @author hzshengxueming
12 | */
13 |
14 | public class HTAnnotatedClass {
15 | /**被标注的类的跳转URL*/
16 | public String[] url;
17 | /**被标注的类的类型信息*/
18 | public TypeElement typeElement;
19 | /**被标注的activity的名称*/
20 | public String activity;
21 | /**退出动画资源id*/
22 | public int exitAnim;
23 | /**进入动画资源id*/
24 | public int entryAnim;
25 |
26 | public HTAnnotatedClass(TypeElement typeElement, String[] url, int entryAnim, int exitAnim) {
27 | this.typeElement = typeElement;
28 | this.activity = typeElement.toString();
29 | this.url = url;
30 | this.entryAnim = entryAnim;
31 | this.exitAnim = exitAnim;
32 | }
33 |
34 | public String getActivity() {
35 | return activity;
36 | }
37 |
38 | public void setActivity(String activity) {
39 | this.activity = activity;
40 | }
41 |
42 | public int getEntryAnim() {
43 | return entryAnim;
44 | }
45 |
46 | public void setEntryAnim(int entryAnim) {
47 | this.entryAnim = entryAnim;
48 | }
49 |
50 | public int getExitAnim() {
51 | return exitAnim;
52 | }
53 |
54 | public void setExitAnim(int exitAnim) {
55 | this.exitAnim = exitAnim;
56 | }
57 |
58 | public TypeElement getTypeElement() {
59 | return typeElement;
60 | }
61 |
62 | public void setTypeElement(TypeElement typeElement) {
63 | this.typeElement = typeElement;
64 | }
65 |
66 | public String[] getUrl() {
67 | return url;
68 | }
69 |
70 | public void setUrl(String[] url) {
71 | this.url = url;
72 | }
73 | }
74 |
--------------------------------------------------------------------------------
/htrouterdispatch-process/src/main/java/com/netease/hearttouch/router/HTClassValidator.java:
--------------------------------------------------------------------------------
1 | /*
2 | * This source code is licensed under the MIT-style license found in the
3 | * LICENSE file in the root directory of this source tree.
4 | */
5 | package com.netease.hearttouch.router;
6 |
7 | import javax.lang.model.element.TypeElement;
8 |
9 | import static javax.lang.model.element.Modifier.ABSTRACT;
10 | import static javax.lang.model.element.Modifier.PUBLIC;
11 |
12 | /**
13 | * @author hzshengxueming
14 | */
15 |
16 | final class HTClassValidator {
17 | /**
18 | * 判断类描述符是否是public的
19 | *
20 | * @param annotatedClass 需要判断的类
21 | * @return 如果是public的返回true,其他返回false
22 | */
23 | static boolean isPublic(TypeElement annotatedClass) {
24 | return annotatedClass.getModifiers().contains(PUBLIC);
25 | }
26 |
27 | /**
28 | * 判断类描述符是否是abstract的
29 | *
30 | * @param annotatedClass 需要判断的类
31 | * @return 如果是abstract的返回true,其他返回false
32 | */
33 | static boolean isAbstract(TypeElement annotatedClass) {
34 | return annotatedClass.getModifiers().contains(ABSTRACT);
35 | }
36 | }
37 |
--------------------------------------------------------------------------------
/htrouterdispatch-process/src/main/java/com/netease/hearttouch/router/HTCodeGenerator.java:
--------------------------------------------------------------------------------
1 | /*
2 | * This source code is licensed under the MIT-style license found in the
3 | * LICENSE file in the root directory of this source tree.
4 | */
5 | package com.netease.hearttouch.router;
6 |
7 | import com.squareup.javapoet.ClassName;
8 | import com.squareup.javapoet.FieldSpec;
9 | import com.squareup.javapoet.MethodSpec;
10 | import com.squareup.javapoet.ParameterizedTypeName;
11 | import com.squareup.javapoet.TypeName;
12 | import com.squareup.javapoet.TypeSpec;
13 |
14 | import java.util.HashMap;
15 | import java.util.LinkedHashMap;
16 | import java.util.List;
17 | import java.util.Map;
18 |
19 | import javax.lang.model.element.Modifier;
20 |
21 | import static com.squareup.javapoet.TypeSpec.classBuilder;
22 | import static javax.lang.model.element.Modifier.ABSTRACT;
23 | import static javax.lang.model.element.Modifier.FINAL;
24 | import static javax.lang.model.element.Modifier.PRIVATE;
25 | import static javax.lang.model.element.Modifier.PUBLIC;
26 | import static javax.lang.model.element.Modifier.STATIC;
27 |
28 | final class HTCodeGenerator {
29 |
30 | private static final String MANAGER_CLASS_NAME = "HTRouterManager";
31 | private static final String LOG_CLASS_NAME = "HTLogUtil";
32 | private static final String ROUTER_HANDLER_CLASS_NAME = "HTRouterHandler";
33 | private static final String ROUTER_ACTIVITY_CLASS_NAME = "HTRouterActivity";
34 | private static final String ROUTER_HANDLER_PARAMS_CLASS_NAME = "HTRouterHandlerParams";
35 |
36 | public static TypeSpec generateManagerClass(String packageName, List classes) {
37 |
38 | TypeSpec.Builder builder = classBuilder(MANAGER_CLASS_NAME)
39 | .addModifiers(PUBLIC, FINAL);
40 | builder.addJavadoc("用于用户启动Activity或者通过URL获得可以跳转的目标\n");
41 | FieldSpec htUrlParamKey = FieldSpec
42 | .builder(String.class, "HT_URL_PARAMS_KEY",
43 | PUBLIC, STATIC, FINAL)
44 | .initializer("\"ht_url_params_map\"").build();
45 | builder.addField(htUrlParamKey);
46 |
47 | FieldSpec mWebActivityClass = FieldSpec
48 | .builder(Class.class, "mWebActivityClass",
49 | Modifier.PRIVATE, Modifier.STATIC)
50 | .build();
51 | builder.addField(mWebActivityClass);
52 | FieldSpec mWebExtraKey = FieldSpec
53 | .builder(String.class, "mWebExtraKey", PRIVATE, STATIC)
54 | .initializer("\"HTUrl\"")
55 | .build();
56 | builder.addField(mWebExtraKey);
57 | FieldSpec NATIVE = FieldSpec
58 | .builder(int.class, "NATIVE", PUBLIC, STATIC, FINAL)
59 | .initializer("1")
60 | .build();
61 | builder.addField(NATIVE);
62 | FieldSpec H5 = FieldSpec
63 | .builder(int.class, "H5", PUBLIC, STATIC, FINAL)
64 | .initializer("2")
65 | .build();
66 | builder.addField(H5);
67 | FieldSpec sHtRouterHandler = FieldSpec
68 | .builder(ClassName.get(packageName, ROUTER_HANDLER_CLASS_NAME), "sHtRouterHandler", PUBLIC, STATIC)
69 | .build();
70 | builder.addField(sHtRouterHandler);
71 | FieldSpec entries = FieldSpec
72 | .builder(ParameterizedTypeName.get(HashMap.class, String.class, HTRouterEntry.class), "entries",
73 | Modifier.PRIVATE, Modifier.STATIC)
74 | .initializer("new $T<>()", TypeName.get(LinkedHashMap.class))
75 | .build();
76 | builder.addField(entries);
77 |
78 | MethodSpec.Builder initMethod = MethodSpec.methodBuilder("init")
79 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
80 | .returns(void.class);
81 | for (HTAnnotatedClass annotatedClass : classes) {
82 | ClassName activity = ClassName.bestGuess(annotatedClass.getActivity());
83 | for (String url : annotatedClass.getUrl()) {
84 | initMethod.addStatement("entries.put($S, new HTRouterEntry($T.class, $S, $L, $L))",
85 | url, activity, url,
86 | annotatedClass.getExitAnim(), annotatedClass.getEntryAnim());
87 | }
88 | }
89 | builder.addMethod(initMethod.build());
90 |
91 |
92 | MethodSpec.Builder setHtRouterHandler = MethodSpec.methodBuilder("setHtRouterHandler")
93 | .addModifiers(PUBLIC, STATIC)
94 | .addParameter(ClassName.get(packageName, ROUTER_HANDLER_CLASS_NAME), "htRouterHandler")
95 | .addStatement("$N = $N", "sHtRouterHandler", "htRouterHandler");
96 | setHtRouterHandler.addJavadoc("设置监听接口,可以在每次进行跳转的时候监听\n");
97 | builder.addMethod(setHtRouterHandler.build());
98 |
99 |
100 | MethodSpec.Builder registerWebActivity = MethodSpec.methodBuilder("registerWebActivity")
101 | .addModifiers(PUBLIC, STATIC)
102 | .addParameter(Class.class, "webActivityClass")
103 | .addStatement("registerWebActivity(webActivityClass, mWebExtraKey)");
104 | registerWebActivity.addJavadoc("设置当前无法处理的URL的降级Webview类,将会通过这个webview进行处理\n" +
105 | "@params webActivityClass 降级处理的webview的类型\n");
106 | builder.addMethod(registerWebActivity.build());
107 |
108 | MethodSpec.Builder registerWebActivity01 = MethodSpec.methodBuilder("registerWebActivity")
109 | .addModifiers(PUBLIC, STATIC)
110 | .addParameter(Class.class, "webActivityClass")
111 | .addParameter(String.class, "webExtraKey")
112 | .beginControlFlow("if(webActivityClass == null || $T.isEmpty(webExtraKey))", ClassName.get("android.text", "TextUtils"))
113 | .addStatement("return")
114 | .endControlFlow()
115 | .addStatement("$N = $N", "mWebActivityClass", "webActivityClass")
116 | .addStatement("$N = $N", "mWebExtraKey", "webExtraKey");
117 | registerWebActivity01.addJavadoc("设置当前无法处理的URL的降级Webview类,将会通过这个webview进行处理\n" +
118 | "@params webActivityClass 降级处理的webview的类型\n" +
119 | "@params webExtraKey 自定义的传递给webview的参数的键\n");
120 | builder.addMethod(registerWebActivity01.build());
121 |
122 | MethodSpec.Builder getSchemaHostAndPathMethod = MethodSpec.methodBuilder("getSchemaHostAndPath")
123 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
124 | .returns(String.class)
125 | .addParameter(String.class, "url")
126 | .beginControlFlow("if($T.isEmpty(url))", ClassName.get("android.text", "TextUtils"))
127 | .addStatement("throw new IllegalArgumentException()")
128 | .endControlFlow()
129 | .addStatement("$T uri = $T.parse(url)", ClassName.get("android.net", "Uri"), ClassName.get("android.net", "Uri"))
130 | .addStatement("return uri.getScheme()+\"://\"+uri.getHost()+uri.getPath()");
131 | builder.addMethod(getSchemaHostAndPathMethod.build());
132 |
133 | MethodSpec.Builder getHostAndPathMethod = MethodSpec.methodBuilder("getHostAndPath")
134 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
135 | .returns(String.class)
136 | .addParameter(String.class, "url")
137 | .beginControlFlow("if($T.isEmpty(url))", ClassName.get("android.text", "TextUtils"))
138 | .addStatement("throw new IllegalArgumentException()")
139 | .endControlFlow()
140 | .addStatement("$T uri = $T.parse(url)", ClassName.get("android.net", "Uri"), ClassName.get("android.net", "Uri"))
141 | .addStatement("return uri.getHost()+uri.getPath()");
142 | builder.addMethod(getHostAndPathMethod.build());
143 |
144 | MethodSpec.Builder getParamsMethod = MethodSpec.methodBuilder("getParams")
145 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
146 | .returns(ParameterizedTypeName.get(HashMap.class, String.class, String.class))
147 | .addParameter(String.class, "url")
148 | .addStatement("$T<$T,$T> params = new $T<>();", ClassName.get(HashMap.class), ClassName.get(String.class), ClassName.get(String.class), ClassName.get(LinkedHashMap.class))
149 | .addStatement("$T uri = $T.parse(url)", ClassName.get("android.net", "Uri"), ClassName.get("android.net", "Uri"))
150 | .addStatement("$T query = uri.getEncodedQuery()", ClassName.get(String.class))
151 | .beginControlFlow("if(query != null)")
152 | .addStatement("$T[] entries = query.split(\"&\")", ClassName.get(String.class))
153 | .beginControlFlow("for($T entry : entries)", ClassName.get(String.class))
154 | .addStatement("$T[] keys = entry.split(\"=\")", ClassName.get(String.class))
155 | .beginControlFlow("if(keys != null && keys.length >= 2)")
156 | .addStatement("params.put(keys[0],keys[1])")
157 | .endControlFlow()
158 | .endControlFlow()
159 | .endControlFlow()
160 | .addStatement("return params");
161 | builder.addMethod(getParamsMethod.build());
162 |
163 |
164 | MethodSpec.Builder getHttpSchemaHostAndPathMethod = MethodSpec.methodBuilder("getHttpSchemaHostAndPath")
165 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
166 | .returns(String.class)
167 | .addParameter(String.class, "url")
168 | .addStatement("$T uri = $T.parse(url)", ClassName.get("android.net", "Uri"), ClassName.get("android.net", "Uri"))
169 | .addStatement("return url.replaceFirst(uri.getScheme(),\"http\")");
170 | builder.addMethod(getHttpSchemaHostAndPathMethod.build());
171 |
172 | MethodSpec.Builder getAnimIdMethod = MethodSpec.methodBuilder("getAnimIdMethod")
173 | .addModifiers(Modifier.PRIVATE, Modifier.STATIC)
174 | .returns(int.class)
175 | .addParameter(Class.class, "activity")
176 | .addParameter(boolean.class, "isExitAnim")
177 | .beginControlFlow("for($T<$T, $T> entry: entries.entrySet())",
178 | Map.Entry.class, String.class, HTRouterEntry.class)
179 | .beginControlFlow("if(entry.getValue().getActivity().toString().equals(activity.toString()))")
180 | .addStatement("return isExitAnim?entry.getValue().getExitAnim():entry.getValue().getEntryAnim()")
181 | .endControlFlow()
182 | .endControlFlow()
183 | .addStatement("return 0");
184 | builder.addMethod(getAnimIdMethod.build());
185 |
186 | MethodSpec.Builder getRouterEntryMethod = MethodSpec.methodBuilder("getRouterEntry")
187 | .addModifiers(Modifier.PRIVATE, Modifier.STATIC)
188 | .addParameter(String.class, "url")
189 | .returns(HTRouterEntry.class)
190 | //.addStatement("$T hostAndPath = getHostAndPath(url)", String.class)
191 | .beginControlFlow("for($T<$T, $T> entry: entries.entrySet())",
192 | Map.Entry.class, String.class, HTRouterEntry.class)
193 | //.beginControlFlow("if(schemaHostAndPath.equals(getSchemaHostAndPath(entry.getValue().getUrl())))")
194 | .beginControlFlow("if(entry.getValue().matches(url))")
195 | .addStatement("return entry.getValue()")
196 | .endControlFlow()
197 | .endControlFlow()
198 | .addStatement("return null");
199 |
200 | builder.addMethod(getRouterEntryMethod.build());
201 |
202 | MethodSpec.Builder findRouterEntryByUrlMethod = MethodSpec.methodBuilder("findRouterEntryByUrl")
203 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
204 | .addParameter(String.class, "url")
205 | .returns(HTRouterEntry.class)
206 | .addStatement("$T hostAndPath = getHostAndPath(url)", String.class)
207 | .beginControlFlow("for($T<$T, $T> entry: entries.entrySet())",
208 | Map.Entry.class, String.class, HTRouterEntry.class)
209 | //.beginControlFlow("if(hostAndPath.equals(getHostAndPath(entry.getValue().getUrl())))")
210 | .beginControlFlow("if(entry.getValue().reverseMatches(url))")
211 | .addStatement("return entry.getValue()")
212 | .endControlFlow()
213 | .endControlFlow()
214 | .addStatement("return null");
215 | findRouterEntryByUrlMethod.addJavadoc("通过URL找到可以跳转的页面的信息\n" +
216 | "@param url 需要进行匹配的URL\n" +
217 | "@return 返回匹配成功后的实体类,如果找不到会返回null\n");
218 | builder.addMethod(findRouterEntryByUrlMethod.build());
219 |
220 | MethodSpec.Builder startActivityMethod = MethodSpec.methodBuilder("startActivity")
221 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
222 | .returns(void.class)
223 | .addParameter(ClassName.get("android.app", "Activity"), "activity")
224 | .addParameter(String.class, "url")
225 | .addParameter(ClassName.get("android.content", "Intent"), "sourceIntent")
226 | .addParameter(boolean.class, "isFinish")
227 | .addStatement("$T intent = null", ClassName.get("android.content", "Intent"))
228 | .addStatement("int exitAnim")
229 | .addStatement("int entryAnim")
230 | .addStatement("$T routerParams = new $T(url, sourceIntent)",
231 | ClassName.get(packageName, "HTRouterHandlerParams"),
232 | ClassName.get(packageName, "HTRouterHandlerParams"))
233 | .beginControlFlow("if (sHtRouterHandler != null && sHtRouterHandler.handleRoute(activity, routerParams))")
234 | .addStatement("return")
235 | .endControlFlow()
236 | .addStatement("url = routerParams.url")
237 | .addStatement("sourceIntent = routerParams.sourceIntent")
238 | .addStatement("HTRouterEntry entry = getRouterEntry(url)")
239 | .beginControlFlow("if(entry != null)")
240 | .addStatement("intent = processIntent(activity, sourceIntent, url, entry.getActivity())")
241 | .addStatement("activity.startActivity(intent)")
242 | .addStatement("exitAnim = $N(activity.getClass(),true)", getAnimIdMethod.build())
243 | .addStatement("entryAnim = $N(entry.getActivity(), false)", getAnimIdMethod.build())
244 | .beginControlFlow("if(isFinish)")
245 | .addStatement("activity.finish()")
246 | .endControlFlow()
247 | .addStatement("activity.overridePendingTransition(entryAnim,exitAnim)")
248 | .nextControlFlow("else if(mWebActivityClass != null)")
249 | .addStatement("intent = processIntent(activity, sourceIntent, url, mWebActivityClass)")
250 | .addStatement("intent.putExtra(mWebExtraKey,getHttpSchemaHostAndPath(url))")
251 | .addStatement("activity.startActivity(intent)")
252 | .addStatement("exitAnim = $N(activity.getClass(),true)", getAnimIdMethod.build())
253 | .addStatement("entryAnim = $N(mWebActivityClass, false)", getAnimIdMethod.build())
254 | .beginControlFlow("if(isFinish)")
255 | .addStatement("activity.finish()")
256 | .endControlFlow()
257 | .addStatement("activity.overridePendingTransition(entryAnim,exitAnim)")
258 | .endControlFlow();
259 | startActivityMethod.addJavadoc("通过URL启动一个页面\n" +
260 | "@param activity 当前需要进行跳转的activity\n" +
261 | "@param url 跳转的目标URL\n" +
262 | "@param sourceIntent 传递进来一个intent,用户数据及启动模式等扩展\n" +
263 | "@param isFinish 跳转后是否需要关闭当前页面\n");
264 | builder.addMethod(startActivityMethod.build());
265 |
266 |
267 | MethodSpec.Builder startActivityForResultMethod = MethodSpec.methodBuilder("startActivityForResult")
268 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
269 | .returns(void.class)
270 | .addParameter(ClassName.get("android.app", "Activity"), "activity")
271 | .addParameter(String.class, "url")
272 | .addParameter(ClassName.get("android.content", "Intent"), "sourceIntent")
273 | .addParameter(boolean.class, "isFinish")
274 | .addParameter(int.class, "requestCode")
275 | .addStatement("$T intent = null", ClassName.get("android.content", "Intent"))
276 | .addStatement("int exitAnim")
277 | .addStatement("int entryAnim")
278 | .addStatement("$T routerParams = new $T(url, sourceIntent)",
279 | ClassName.get(packageName, "HTRouterHandlerParams"),
280 | ClassName.get(packageName, "HTRouterHandlerParams"))
281 | .beginControlFlow("if (sHtRouterHandler != null && sHtRouterHandler.handleRoute(activity, routerParams))")
282 | .addStatement("return")
283 | .endControlFlow()
284 | .addStatement("url = routerParams.url")
285 | .addStatement("sourceIntent = routerParams.sourceIntent")
286 | .addStatement("HTRouterEntry entry = getRouterEntry(url)")
287 | .beginControlFlow("if(entry != null)")
288 | .addStatement("intent = processIntent(activity, sourceIntent, url, entry.getActivity())")
289 | .addStatement("activity.startActivityForResult(intent,requestCode)")
290 | .addStatement("exitAnim = $N(activity.getClass(),true)", getAnimIdMethod.build())
291 | .addStatement("entryAnim = $N(entry.getActivity(), false)", getAnimIdMethod.build())
292 | .beginControlFlow("if(isFinish)")
293 | .addStatement("activity.finish()")
294 | .endControlFlow()
295 | .addStatement("activity.overridePendingTransition(entryAnim,exitAnim)")
296 | .nextControlFlow("else if(mWebActivityClass != null)")
297 | .addStatement("intent = processIntent(activity, sourceIntent, url, mWebActivityClass)")
298 | .addStatement("intent.putExtra(mWebExtraKey,getHttpSchemaHostAndPath(url))")
299 | .addStatement("activity.startActivityForResult(intent,requestCode)")
300 | .addStatement("exitAnim = $N(activity.getClass(),true)", getAnimIdMethod.build())
301 | .addStatement("entryAnim = $N(mWebActivityClass, false)", getAnimIdMethod.build())
302 | .beginControlFlow("if(isFinish)")
303 | .addStatement("activity.finish()")
304 | .endControlFlow()
305 | .addStatement("activity.overridePendingTransition(entryAnim,exitAnim)")
306 | .endControlFlow();
307 | startActivityForResultMethod.addJavadoc("通过URL启动一个页面,同时可以获得result回调\n" +
308 | "@param activity 当前需要进行跳转的activity\n" +
309 | "@param url 跳转的目标URL\n" +
310 | "@param sourceIntent 传递进来一个intent,用户数据及启动模式等扩展\n" +
311 | "@param isFinish 跳转后是否需要关闭当前页面\n");
312 | builder.addMethod(startActivityForResultMethod.build());
313 |
314 |
315 | MethodSpec.Builder startActivityForResultFragmentMethod = MethodSpec.methodBuilder("startActivityForResult")
316 | .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
317 | .returns(void.class)
318 | .addParameter(ClassName.get("android.app", "Fragment"), "fragment")
319 | .addParameter(String.class, "url")
320 | .addParameter(ClassName.get("android.content", "Intent"), "sourceIntent")
321 | .addParameter(boolean.class, "isFinish")
322 | .addParameter(int.class, "requestCode")
323 | .addStatement("$T intent = null", ClassName.get("android.content", "Intent"))
324 | .addStatement("int exitAnim")
325 | .addStatement("int entryAnim")
326 | .addStatement("$T routerParams = new $T(url, sourceIntent)",
327 | ClassName.get(packageName, "HTRouterHandlerParams"),
328 | ClassName.get(packageName, "HTRouterHandlerParams"))
329 | .beginControlFlow("if (sHtRouterHandler != null && sHtRouterHandler.handleRoute(fragment.getActivity(), routerParams))")
330 | .addStatement("return")
331 | .endControlFlow()
332 | .addStatement("url = routerParams.url")
333 | .addStatement("sourceIntent = routerParams.sourceIntent")
334 | .addStatement("HTRouterEntry entry = getRouterEntry(url)")
335 | .beginControlFlow("if(entry != null)")
336 | .addStatement("intent = processIntent(fragment.getActivity(), sourceIntent, url, entry.getActivity())")
337 | .addStatement("fragment.startActivityForResult(intent,requestCode)")
338 | .addStatement("exitAnim = $N(fragment.getActivity().getClass(),true)", getAnimIdMethod.build())
339 | .addStatement("entryAnim = $N(entry.getActivity(), false)", getAnimIdMethod.build())
340 | .beginControlFlow("if(isFinish)")
341 | .addStatement("fragment.getActivity().finish()")
342 | .endControlFlow()
343 | .addStatement("fragment.getActivity().overridePendingTransition(entryAnim,exitAnim)")
344 | .nextControlFlow("else if(mWebActivityClass != null)")
345 | .addStatement("intent = processIntent(fragment.getActivity(), sourceIntent, url, mWebActivityClass)")
346 | .addStatement("intent.putExtra(mWebExtraKey,getHttpSchemaHostAndPath(url))")
347 | .addStatement("fragment.startActivityForResult(intent,requestCode)")
348 | .addStatement("exitAnim = $N(fragment.getActivity().getClass(),true)", getAnimIdMethod.build())
349 | .addStatement("entryAnim = $N(mWebActivityClass, false)", getAnimIdMethod.build())
350 | .beginControlFlow("if(isFinish)")
351 | .addStatement("fragment.getActivity().finish()")
352 | .endControlFlow()
353 | .addStatement("fragment.getActivity().overridePendingTransition(entryAnim,exitAnim)")
354 | .endControlFlow();
355 | startActivityForResultFragmentMethod.addJavadoc("通过URL启动一个页面,同时可以获得result回调,回调在fragment中\n" +
356 | "@param fragment 当前需要进行跳转的fragment\n" +
357 | "@param url 跳转的目标URL\n" +
358 | "@param sourceIntent 传递进来一个intent,用户数据及启动模式等扩展\n" +
359 | "@param isFinish 跳转后是否需要关闭当前页面\n");
360 | builder.addMethod(startActivityForResultFragmentMethod.build());
361 |
362 | MethodSpec.Builder setDebugMethod = MethodSpec.methodBuilder("setDebugMode")
363 | .addModifiers(PUBLIC, STATIC)
364 | .returns(void.class)
365 | .addParameter(boolean.class, "debug")
366 | .addStatement("$T.setDebugMode(debug)", ClassName.get(packageName, "HTLogUtil"));
367 | builder.addMethod(setDebugMethod.build());
368 |
369 |
370 | MethodSpec.Builder processIntentMethod = MethodSpec.methodBuilder("processIntent")
371 | .addModifiers(PRIVATE, STATIC)
372 | .returns(ClassName.get("android.content", "Intent"))
373 | .addParameter(ClassName.get("android.content", "Context"), "context")
374 | .addParameter(ClassName.get("android.content", "Intent"), "sourceIntent")
375 | .addParameter(String.class, "url")
376 | .addParameter(Class.class, "activityClass")
377 | .addStatement("$T intent = null", ClassName.get("android.content", "Intent"))
378 | .beginControlFlow("if(sourceIntent != null)")
379 | .addStatement("intent = ($T)sourceIntent.clone()", ClassName.get("android.content", "Intent"))
380 | .addStatement("intent.setClass(context, activityClass)")
381 | .nextControlFlow("else")
382 | .addStatement("intent = new Intent(context, activityClass)")
383 | .endControlFlow()
384 | .addStatement("intent.setData($T.parse(url))", ClassName.get("android.net", "Uri"))
385 | .addStatement("HashMap paramsMap = getParams(url)")
386 | .beginControlFlow("if(paramsMap != null)")
387 | .addStatement("intent.putExtra(HT_URL_PARAMS_KEY, paramsMap)")
388 | .endControlFlow()
389 | .addStatement("return intent");
390 | builder.addMethod(processIntentMethod.build());
391 |
392 | return builder.build();
393 | }
394 |
395 |
396 | public static TypeSpec generateHTLogUtilClass(List classes) {
397 | TypeSpec.Builder builder = classBuilder(LOG_CLASS_NAME)
398 | .addModifiers(PUBLIC, FINAL);
399 |
400 | FieldSpec tag = FieldSpec
401 | .builder(String.class, "TAG", PRIVATE, STATIC, FINAL)
402 | .initializer("\"HT\"")
403 | .build();
404 | builder.addField(tag);
405 |
406 | FieldSpec sDebug = FieldSpec
407 | .builder(boolean.class, "sDebug", PRIVATE, STATIC)
408 | .initializer("false")
409 | .build();
410 | builder.addField(sDebug);
411 |
412 | MethodSpec.Builder setDugMode = MethodSpec.methodBuilder("setDebugMode")
413 | .addModifiers(PUBLIC, STATIC)
414 | .addParameter(boolean.class, "debug")
415 | .addStatement("$N = $N", "sDebug", "debug");
416 | builder.addMethod(setDugMode.build());
417 |
418 | MethodSpec.Builder d = MethodSpec.methodBuilder("d")
419 | .addModifiers(PUBLIC, STATIC)
420 | .addParameter(String.class, "message")
421 | .beginControlFlow("if(sDebug)")
422 | .addStatement("$T.d(TAG,message)", ClassName.get("android.util", "Log"))
423 | .endControlFlow();
424 | builder.addMethod(d.build());
425 |
426 | return builder.build();
427 | }
428 |
429 | public static TypeSpec generateHTRouterHandlerClass(String packageName, List classes) {
430 | TypeSpec.Builder builder = TypeSpec.interfaceBuilder(ROUTER_HANDLER_CLASS_NAME)
431 | .addModifiers(PUBLIC);
432 | MethodSpec.Builder handleMethod = MethodSpec.methodBuilder("handleRoute")
433 | .addModifiers(PUBLIC, ABSTRACT)
434 | .returns(boolean.class)
435 | .addParameter(ClassName.get("android.content", "Context"), "context")
436 | .addParameter(ClassName.get(packageName, "HTRouterHandlerParams"), "routerParams");
437 | // .addParameter(String.class, "url")
438 | // .addParameter(ClassName.get("android.content", "Intent"), "intent")
439 | // .addParameter(int.class, "type");
440 | builder.addMethod(handleMethod.build());
441 | return builder.build();
442 | }
443 |
444 | public static TypeSpec generateHTRouterActivityClass(String packageName, List classes) {
445 | TypeSpec.Builder builder = TypeSpec.classBuilder(ROUTER_ACTIVITY_CLASS_NAME)
446 | .addModifiers(Modifier.PUBLIC)
447 | .superclass(ClassName.get("android.app", "Activity"));
448 |
449 | MethodSpec.Builder onCreateMethod = MethodSpec.methodBuilder("onCreate")
450 | .addModifiers(Modifier.PROTECTED)
451 | .addAnnotation(Override.class)
452 | .returns(void.class)
453 | .addParameter(ClassName.get("android.os", "Bundle"), "savedInstanceState")
454 | .addStatement("super.onCreate(savedInstanceState)")
455 | .addStatement("$T intent = getIntent()", ClassName.get("android.content", "Intent"))
456 | .beginControlFlow("if (intent != null && intent.getData()!= null)")
457 | .addStatement("$T.d(\"receive URL:\" + intent.getData().toString())", ClassName.get(packageName, "HTLogUtil"))
458 | .addStatement("$T url = intent.getData().toString()", String.class)
459 | .addStatement("$T.startActivity($T.this,url,intent,true)", ClassName.get(packageName, "HTRouterManager"), ClassName.get(packageName, "HTRouterActivity"))
460 | .nextControlFlow("else")
461 | .addStatement("finish()")
462 | .addStatement("$T.d(\"page error,needs URL format \")", ClassName.get(packageName, "HTLogUtil"))
463 | .addStatement("$T.makeText($T.this,\"page error\",Toast.LENGTH_SHORT).show()", ClassName.get("android.widget", "Toast"), ClassName.get(packageName, "HTRouterActivity"))
464 | .endControlFlow();
465 | builder.addMethod(onCreateMethod.build());
466 | return builder.build();
467 | }
468 |
469 | public static TypeSpec generateHTRouterHandlerParamsClass(String packageName, List classes) {
470 | TypeSpec.Builder builder = TypeSpec.classBuilder(ROUTER_HANDLER_PARAMS_CLASS_NAME)
471 | .addModifiers(Modifier.PUBLIC);
472 |
473 | FieldSpec urlFiled = FieldSpec
474 | .builder(String.class, "url", PUBLIC)
475 | .build();
476 | builder.addField(urlFiled);
477 |
478 | FieldSpec intentFiled = FieldSpec
479 | .builder(ClassName.get("android.content", "Intent"), "sourceIntent", PUBLIC)
480 | .build();
481 | builder.addField(intentFiled);
482 |
483 | MethodSpec.Builder constructorMethod = MethodSpec.constructorBuilder()
484 | .addModifiers(PUBLIC)
485 | .addParameter(String.class, "url")
486 | .addParameter(ClassName.get("android.content", "Intent"), "sourceIntent")
487 | .addStatement("this.url = url")
488 | .addStatement("this.sourceIntent = sourceIntent");
489 | builder.addMethod(constructorMethod.build());
490 |
491 | // MethodSpec.Builder getUrlMethod = MethodSpec.methodBuilder("getUrl")
492 | // .addModifiers(PUBLIC)
493 | // .returns(String.class)
494 | // .addStatement("return mUrl");
495 | // builder.addMethod(getUrlMethod.build());
496 | //
497 | // MethodSpec.Builder setUrlMethod = MethodSpec.methodBuilder("setUrl")
498 | // .addModifiers(PUBLIC)
499 | // .returns(void.class)
500 | // .addParameter(String.class, "url")
501 | // .addStatement("mUrl = url");
502 | // builder.addMethod(setUrlMethod.build());
503 | //
504 | // MethodSpec.Builder getTypeMethod = MethodSpec.methodBuilder("getType")
505 | // .addModifiers(PUBLIC)
506 | // .returns(String.class)
507 | // .addStatement("return mUrl");
508 | // builder.addMethod(getTypeMethod.build());
509 | //
510 | // MethodSpec.Builder setTypeMethod = MethodSpec.methodBuilder("setType")
511 | // .addModifiers(PUBLIC)
512 | // .returns(void.class)
513 | // .addParameter(int.class, "type")
514 | // .addStatement("mType = type");
515 | // builder.addMethod(setUrlMethod.build());
516 | //
517 | // MethodSpec.Builder getIntentMethod = MethodSpec.methodBuilder("getIntent")
518 | // .addModifiers(PUBLIC)
519 | // .returns(ClassName.get("android.content", "Intent"))
520 | // .addStatement("return sourceIntent");
521 | // builder.addMethod(getUrlMethod.build());
522 | //
523 | // MethodSpec.Builder setUrlMethod = MethodSpec.methodBuilder("setUrl")
524 | // .addModifiers(PUBLIC)
525 | // .returns(void.class)
526 | // .addParameter(String.class, "url")
527 | // .addStatement("mUrl = url");
528 | // builder.addMethod(setUrlMethod.build());
529 | return builder.build();
530 | }
531 | }
532 |
--------------------------------------------------------------------------------
/htrouterdispatch-process/src/main/java/com/netease/hearttouch/router/HTRouterDispatchProcess.java:
--------------------------------------------------------------------------------
1 | /*
2 | * This source code is licensed under the MIT-style license found in the
3 | * LICENSE file in the root directory of this source tree.
4 | */
5 | package com.netease.hearttouch.router;
6 |
7 | import com.google.auto.service.AutoService;
8 | import com.squareup.javapoet.JavaFile;
9 | import com.squareup.javapoet.TypeSpec;
10 |
11 | import java.io.IOException;
12 | import java.util.ArrayList;
13 | import java.util.HashMap;
14 | import java.util.List;
15 | import java.util.Map;
16 | import java.util.Set;
17 |
18 | import javax.annotation.processing.AbstractProcessor;
19 | import javax.annotation.processing.Filer;
20 | import javax.annotation.processing.Messager;
21 | import javax.annotation.processing.ProcessingEnvironment;
22 | import javax.annotation.processing.Processor;
23 | import javax.annotation.processing.RoundEnvironment;
24 | import javax.lang.model.SourceVersion;
25 | import javax.lang.model.element.Element;
26 | import javax.lang.model.element.TypeElement;
27 |
28 | import static com.squareup.javapoet.JavaFile.builder;
29 | import static java.util.Collections.singleton;
30 | import static javax.lang.model.SourceVersion.latestSupported;
31 | import static javax.tools.Diagnostic.Kind.ERROR;
32 |
33 |
34 | /**
35 | * 识别和预处理注解的类,会在编译期生成代码
36 | */
37 | @AutoService(Processor.class)
38 | public class HTRouterDispatchProcess extends AbstractProcessor {
39 | private Messager messager;
40 | private Filer filer;
41 | private String packageName = "com.netease.hearttouch.router";
42 | private static final String ANNOTATION = "@" + HTRouter.class.getSimpleName();
43 | private static final int CODE_TYPE_HT_LOG_UTIL = 0;
44 | private static final int CODE_TYPE_HT_ROUTER_HANDLER = 1;
45 | private static final int CODE_TYPE_HT_ROUTER_MANAGER = 2;
46 | private static final int CODE_TYPE_HT_ROUTER_ACTIVITY = 3;
47 | private static final int CODE_TYPE_HT_ROUTER_HANDLER_PARAMS = 4;
48 | private static final Map CODE_TYPE_MAP= new HashMap() {
49 | {
50 | put(CODE_TYPE_HT_LOG_UTIL, "Couldn't generate HTLogUtil class");
51 | put(CODE_TYPE_HT_ROUTER_HANDLER, "Couldn't generate HTRouterListener class");
52 | put(CODE_TYPE_HT_ROUTER_MANAGER, "Couldn't generate HTRouterManager class");
53 | put(CODE_TYPE_HT_ROUTER_ACTIVITY, "Couldn't generate HTRouterActivity class");
54 | put(CODE_TYPE_HT_ROUTER_HANDLER_PARAMS, "Couldn't generate HTRouterHandlerParams class");
55 | }
56 | };
57 |
58 | @Override
59 | public synchronized void init(ProcessingEnvironment processingEnv) {
60 | super.init(processingEnv);
61 | messager = processingEnv.getMessager();
62 | filer = processingEnv.getFiler();
63 | }
64 |
65 | @Override
66 | public Set getSupportedAnnotationTypes() {
67 | return singleton(HTRouter.class.getCanonicalName());
68 | }
69 |
70 | @Override
71 | public SourceVersion getSupportedSourceVersion() {
72 | return latestSupported();
73 | }
74 |
75 | /**
76 | * 主要的处理注解的类,会拿到所有的注解相关的类
77 | *
78 | * @param annotations
79 | * @param roundEnv
80 | * @return 处理成功返回true
81 | */
82 | @Override
83 | public boolean process(Set extends TypeElement> annotations, RoundEnvironment roundEnv) {
84 | List annotatedClasses = new ArrayList<>();
85 | //获取所有通过HTRouter注解的项,遍历
86 | for (Element annotatedElement : roundEnv.getElementsAnnotatedWith(HTRouter.class)) {
87 | TypeElement annotatedClass = (TypeElement) annotatedElement;
88 | //检测是否是支持的注解类型,如果不是里面会报错
89 | if (!isValidClass(annotatedClass)) {
90 | return true;
91 | }
92 | //获取到信息,把注解类的信息加入到列表中
93 | HTRouter htRouter = annotatedElement.getAnnotation(HTRouter.class);
94 | annotatedClasses.add(new HTAnnotatedClass(annotatedClass, htRouter.url(), htRouter.entryAnim(), htRouter.exitAnim()));
95 | }
96 |
97 | //生成各种类
98 | for (Integer type : CODE_TYPE_MAP.keySet()){
99 | try {
100 | generateCode(annotatedClasses, type);
101 | } catch (IOException e) {
102 | messager.printMessage(ERROR, CODE_TYPE_MAP.get(type));
103 | }
104 | }
105 | return true;
106 | }
107 |
108 | private boolean isValidClass(TypeElement annotatedClass) {
109 |
110 | if (!HTClassValidator.isPublic(annotatedClass)) {
111 | String message = String.format("Classes annotated with %s must be public.", ANNOTATION);
112 | messager.printMessage(ERROR, message, annotatedClass);
113 | return false;
114 | }
115 |
116 | if (HTClassValidator.isAbstract(annotatedClass)) {
117 | String message = String.format("Classes annotated with %s must not be abstract.", ANNOTATION);
118 | messager.printMessage(ERROR, message, annotatedClass);
119 | return false;
120 | }
121 |
122 |
123 | return true;
124 | }
125 |
126 | private void generateCode(List annotatedClasses, int type) throws IOException {
127 | if (annotatedClasses != null && annotatedClasses.size() == 0) {
128 | return;
129 | }
130 | TypeSpec generatedClass = null;
131 | switch (type){
132 | case CODE_TYPE_HT_LOG_UTIL:
133 | generatedClass = HTCodeGenerator.generateHTLogUtilClass(annotatedClasses);
134 | break;
135 | case CODE_TYPE_HT_ROUTER_HANDLER:
136 | generatedClass = HTCodeGenerator.generateHTRouterHandlerClass(packageName, annotatedClasses);
137 | break;
138 | case CODE_TYPE_HT_ROUTER_MANAGER:
139 | generatedClass = HTCodeGenerator.generateManagerClass(packageName, annotatedClasses);
140 | break;
141 | case CODE_TYPE_HT_ROUTER_ACTIVITY:
142 | generatedClass = HTCodeGenerator.generateHTRouterActivityClass(packageName, annotatedClasses);
143 | break;
144 | case CODE_TYPE_HT_ROUTER_HANDLER_PARAMS:
145 | generatedClass = HTCodeGenerator.generateHTRouterHandlerParamsClass(packageName, annotatedClasses);
146 | break;
147 | default:
148 | messager.printMessage(ERROR, "unsupported code type.");
149 | return;
150 | }
151 | JavaFile javaFile = builder(packageName, generatedClass).build();
152 | javaFile.writeTo(filer);
153 | }
154 | }
155 |
--------------------------------------------------------------------------------
/htrouterdispatch/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/htrouterdispatch/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'java'
2 |
3 | sourceCompatibility = JavaVersion.VERSION_1_7
4 | targetCompatibility = JavaVersion.VERSION_1_7
5 |
6 | dependencies {
7 |
8 | }
--------------------------------------------------------------------------------
/htrouterdispatch/src/main/java/com/netease/hearttouch/router/HTRouter.java:
--------------------------------------------------------------------------------
1 | /*
2 | * This source code is licensed under the MIT-style license found in the
3 | * LICENSE file in the root directory of this source tree.
4 | */
5 | package com.netease.hearttouch.router;
6 |
7 | import java.lang.annotation.ElementType;
8 | import java.lang.annotation.Retention;
9 | import java.lang.annotation.RetentionPolicy;
10 | import java.lang.annotation.Target;
11 |
12 | @Target(ElementType.TYPE)
13 | @Retention(RetentionPolicy.RUNTIME)
14 | public @interface HTRouter {
15 | /** activity对应url */
16 | String[] url();
17 |
18 | /** Activity进场动画资源 */
19 | int entryAnim() default 0;
20 |
21 | /** Activity出场动画资源 */
22 | int exitAnim() default 0;
23 | }
24 |
--------------------------------------------------------------------------------
/htrouterdispatch/src/main/java/com/netease/hearttouch/router/HTRouterEntry.java:
--------------------------------------------------------------------------------
1 | /*
2 | * This source code is licensed under the MIT-style license found in the
3 | * LICENSE file in the root directory of this source tree.
4 | */
5 | package com.netease.hearttouch.router;
6 |
7 | import java.net.URLEncoder;
8 | import java.util.regex.Pattern;
9 |
10 | /**
11 | * 保存每个被标注的Activity的信息,包括可以匹配的URL,进入及退出动画等
12 | * 主要提供匹配URL功能,判断目标URL是否可以跳转到当前Activity
13 | *
14 | * @author hzshengxueming
15 | */
16 |
17 | public class HTRouterEntry {
18 | /** URL参数值对应的正则表达式 */
19 | private static final String PARAM_VALUE = "([a-zA-Z0-9_#'!+%~,\\-\\.\\$]*)";
20 | /** URL参数键对应的正则表达式 */
21 | private static final String PARAM = "([a-zA-Z][a-zA-Z0-9_-]*)";
22 | /** 填充字段的正则,用于将{id}转换为实际匹配的正则表达式 */
23 | private static final String PARAM_REGEX = "%7B(" + PARAM + ")%7D";
24 | /** 正向匹配正则 */
25 | private final Pattern regex;
26 | /** 逆向匹配正则 */
27 | private final Pattern regexReverse;
28 | /** 跳转页面的类型信息 */
29 | private Class> activity;
30 | /** 本页面可以匹配的URL */
31 | private String url;
32 | /** 退场动画 */
33 | private int exitAnim;
34 | /** 进场动画 */
35 | private int entryAnim;
36 |
37 | /**
38 | * 构造一个用于保存URL与页面对应关系的类
39 | *
40 | * @param activity 目标页面的类型信息
41 | * @param url 用于匹配的URL信息
42 | * @param exitAnim 退出动画的资源id
43 | * @param entryAnim 进入动画的资源id
44 | */
45 | public HTRouterEntry(Class> activity, String url, int exitAnim, int entryAnim) {
46 | this.activity = activity;
47 | this.url = url;
48 | this.exitAnim = exitAnim;
49 | this.entryAnim = entryAnim;
50 | //替换掉URL中填充的占位信息,例如{id}
51 | this.regex = Pattern.compile(hostAndPath(url).replaceAll(PARAM_REGEX, PARAM_VALUE) + "$");
52 | this.regexReverse = Pattern.compile(hostAndPath(url).replaceAll(PARAM_REGEX, PARAM_VALUE) + "$");
53 | }
54 |
55 | /**
56 | * 构造一个用于保存URL与页面对应关系的类,默认进出场动画
57 | *
58 | * @param activity 目标页面的类型信息
59 | * @param url 用于匹配的URL信息
60 | */
61 | public HTRouterEntry(Class> activity, String url) {
62 | this.activity = activity;
63 | this.url = url;
64 | this.exitAnim = 0;
65 | this.entryAnim = 0;
66 | //替换掉URL中填充的占位信息,例如{id}
67 | this.regex = Pattern.compile(hostAndPath(url).replaceAll(PARAM_REGEX, PARAM_VALUE) + "$");
68 | this.regexReverse = Pattern.compile(hostAndPath(url).replaceAll(PARAM_REGEX, PARAM_VALUE) + "$");
69 | }
70 |
71 | /**
72 | * 进行正向匹配,判断传入的URL是否能跳转到当前页面
73 | *
74 | * @param inputUrl 需要进行判断的URL
75 | * @return 如果能跳转返回true,如果不能则返回false
76 | */
77 | public boolean matches(String inputUrl) {
78 | return inputUrl != null && regex.matcher(hostAndPath(inputUrl)).find();
79 | }
80 |
81 | /**
82 | * 进行反向匹配,判断传入的URL是否能跳转到当前页面,预留接口
83 | * 目的是忽略scheme进行匹配
84 | *
85 | * @param inputUrl 需要进行判断的URL
86 | * @return 如果能跳转返回true,如果不能则返回false
87 | */
88 | public boolean reverseMatches(String inputUrl) {
89 | return inputUrl != null && regexReverse.matcher(hostAndPath(inputUrl)).find();
90 | }
91 |
92 |
93 | private String hostAndPath(String url) {
94 | int postion = url.indexOf("://");
95 | //去掉scheme
96 | if (postion != -1) {
97 | url = url.substring(postion + "://".length());
98 | }
99 | //去掉参数
100 | postion = url.indexOf("?");
101 | if(postion != -1) {
102 | url = url.substring(0, postion);
103 | }
104 | String[] urls = url.split("/");
105 | //每一段都encode一下
106 | StringBuilder sb = new StringBuilder();
107 | for (int i = 0; i < urls.length; i++) {
108 | sb.append(URLEncoder.encode(urls[i]));
109 | if (i != urls.length - 1){
110 | sb.append('/');
111 | }
112 | }
113 | return sb.toString();
114 |
115 | }
116 |
117 | public Class> getActivity() {
118 | return activity;
119 | }
120 |
121 | public void setActivity(Class> activity) {
122 | this.activity = activity;
123 | }
124 |
125 | public int getEntryAnim() {
126 | return entryAnim;
127 | }
128 |
129 | public void setEntryAnim(int entryAnim) {
130 | this.entryAnim = entryAnim;
131 | }
132 |
133 | public int getExitAnim() {
134 | return exitAnim;
135 | }
136 |
137 | public void setExitAnim(int exitAnim) {
138 | this.exitAnim = exitAnim;
139 | }
140 |
141 | public String getUrl() {
142 | return url;
143 | }
144 |
145 | public void setUrl(String url) {
146 | this.url = url;
147 | }
148 | }
149 |
--------------------------------------------------------------------------------
/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':demo', ':htrouterdispatch', ':htrouterdispatch-process'
2 |
--------------------------------------------------------------------------------