├── .gitignore ├── CONTRIBUTING.md ├── ISSUE_TEMPLATE.md ├── LICENSE ├── README.md ├── README_EN.md ├── app ├── .gitignore ├── build.gradle ├── libs │ └── tbs_sdk_thirdapp_v3.6.0.1315_43612_sharewithdownload_withoutGame_obfs_20180718_102847.jar ├── proguard-rules.pro └── src │ └── main │ ├── AndroidManifest.xml │ ├── assets │ └── static │ │ └── scripts │ │ └── jquery.min.js │ ├── java │ └── ren │ │ └── yale │ │ └── android │ │ └── cachewebview │ │ ├── App.java │ │ ├── JSInterface.java │ │ ├── MainActivity.java │ │ ├── WebResourceRequestAdapter.java │ │ ├── WebResourceResponseAdapter.java │ │ ├── X5Activity.java │ │ └── x5 │ │ └── X5WebView.java │ └── res │ ├── layout │ ├── activity_main.xml │ ├── activity_main2.xml │ ├── activity_main3.xml │ └── activity_x5.xml │ ├── mipmap-hdpi │ ├── ic_launcher.png │ └── ic_launcher_round.png │ ├── mipmap-mdpi │ ├── ic_launcher.png │ └── ic_launcher_round.png │ ├── mipmap-xhdpi │ ├── ic_launcher.png │ └── ic_launcher_round.png │ ├── mipmap-xxhdpi │ ├── ic_launcher.png │ └── ic_launcher_round.png │ ├── mipmap-xxxhdpi │ ├── ic_launcher.png │ └── ic_launcher_round.png │ └── values │ ├── arrays.xml │ ├── colors.xml │ ├── strings.xml │ └── styles.xml ├── art └── assets.png ├── build.gradle ├── cachewebviewlib ├── .gitignore ├── build.gradle ├── config │ └── bintray.gradle ├── proguard-rules.pro └── src │ └── main │ ├── AndroidManifest.xml │ ├── java │ └── ren │ │ └── yale │ │ └── android │ │ └── cachewebviewlib │ │ ├── AssetsLoader.java │ │ ├── CacheType.java │ │ ├── CacheWebViewLog.java │ │ ├── DynamicCacheLoader.java │ │ ├── HttpCacheInterceptor.java │ │ ├── ResourceInterceptor.java │ │ ├── WebViewCacheInterceptor.java │ │ ├── WebViewCacheInterceptorInst.java │ │ ├── WebViewRequestInterceptor.java │ │ ├── config │ │ └── CacheExtensionConfig.java │ │ └── utils │ │ ├── AppUtils.java │ │ ├── FileUtil.java │ │ ├── MD5Utils.java │ │ ├── MimeTypeMapUtils.java │ │ ├── NetUtils.java │ │ ├── OKHttpFile.java │ │ └── TimeUtils.java │ └── res │ └── values │ └── strings.xml ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /.idea 4 | /local.properties 5 | .DS_Store 6 | /build 7 | /captures 8 | .externalNativeBuild 9 | *.apk 10 | *.jks -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | - fork https://github.com/yale8848/CacheWebView.git 2 | - git checkout **`dev-2.0`** 3 | - commit & push 4 | - pull request 5 | -------------------------------------------------------------------------------- /ISSUE_TEMPLATE.md: -------------------------------------------------------------------------------- 1 | Mobile: 2 | 3 | Android OS: 4 | 5 | NetWork: 6 | 7 | URL: 8 | 9 | LIB Version: 10 | 11 | --- 12 | 13 | 14 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2017 Yale Ren 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. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # CacheWebView 2 | 3 | [![](https://img.shields.io/badge/jcenter-2.1.8-519dd9.svg)](https://bintray.com/yale8848/maven/CacheWebView/2.1.8) 4 | 5 | [English](https://github.com/yale8848/CacheWebView/blob/master/README_EN.md) 6 | 7 | CacheWebView通过拦截资源实现自定义缓存静态资源。突破WebView缓存空间限制,让缓存更简单。让网站离线也能正常访问。 8 | 9 | ## 为什么要用CacheWebView 10 | 11 | - 让WebView缓存空间更大 12 | - 强制缓存静态资源,这样会更快 13 | - 想方便的拿到web缓存资源,比如说从缓存中拿页面已经加载过的图片 14 | 15 | 16 | ## 使用方式 17 | 18 | ### 引入库 19 | 20 | **注意2.x.x 不兼容 1.x.x** 21 | 22 | ```groovy 23 | implementation 'ren.yale.android:cachewebviewlib:2.2.1' 24 | ``` 25 | 26 | ### 修改代码 27 | 28 | Application 里初始化 29 | 30 | ``` 31 | 32 | WebViewCacheInterceptorInst.getInstance(). 33 | init(new WebViewCacheInterceptor.Builder(this)); 34 | 35 | ``` 36 | 37 | 38 | 给WebView添加拦截 39 | 40 | - 如果你的项目minSdkVersion>=21 41 | 42 | ``` 43 | mWebView.setWebViewClient(new WebViewClient(){ 44 | 45 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 46 | @Nullable 47 | @Override 48 | public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) { 49 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(request); 50 | } 51 | 52 | @Nullable 53 | @Override 54 | public WebResourceResponse shouldInterceptRequest(WebView view, String url) { 55 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(url); 56 | } 57 | }); 58 | 59 | ``` 60 | 61 | - 如果你的项目minSdkVersion<21 62 | 63 | 将调用 `mWebView.loadUrl(url)` 的地方替换为:`WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,url)` 64 | 65 | ``` 66 | 67 | mWebView.setWebViewClient(new WebViewClient(){ 68 | 69 | 70 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 71 | @Override 72 | public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) { 73 | WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,request.getUrl().toString()); 74 | return true; 75 | } 76 | 77 | @Override 78 | public boolean shouldOverrideUrlLoading(WebView view, String url) { 79 | WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,url); 80 | return true; 81 | } 82 | 83 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 84 | @Nullable 85 | @Override 86 | public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) { 87 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(request); 88 | } 89 | 90 | @Nullable 91 | @Override 92 | public WebResourceResponse shouldInterceptRequest(WebView view, String url) { 93 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(url); 94 | } 95 | }); 96 | 97 | ``` 98 | 99 | 以上就配置完毕,其他代码不用改,这样拥有默认100M缓存空间;如果你需要更详细的配置,可以看看下面的进阶设置; 100 | 101 | --- 102 | 103 | - 腾讯X5内核WebView兼容处理 104 | 105 | 106 | ``` 107 | mWebView.setWebViewClient(new WebViewClient() { 108 | 109 | @Override 110 | public WebResourceResponse shouldInterceptRequest(WebView webView, String s) { 111 | return WebResourceResponseAdapter.adapter(WebViewCacheInterceptorInst.getInstance(). 112 | interceptRequest(s)); 113 | } 114 | 115 | @Override 116 | public WebResourceResponse shouldInterceptRequest(WebView webView, WebResourceRequest webResourceRequest) { 117 | 118 | return WebResourceResponseAdapter.adapter(WebViewCacheInterceptorInst.getInstance(). 119 | interceptRequest(WebResourceRequestAdapter.adapter(webResourceRequest))); 120 | } 121 | }); 122 | 123 | ``` 124 | 125 | 下面是兼容代码,可以参考: 126 | 127 | ``` 128 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 129 | public class WebResourceRequestAdapter implements android.webkit.WebResourceRequest { 130 | 131 | private com.tencent.smtt.export.external.interfaces.WebResourceRequest mWebResourceRequest; 132 | 133 | private WebResourceRequestAdapter(com.tencent.smtt.export.external.interfaces.WebResourceRequest x5Request){ 134 | mWebResourceRequest = x5Request; 135 | } 136 | 137 | public static WebResourceRequestAdapter adapter(com.tencent.smtt.export.external.interfaces.WebResourceRequest x5Request){ 138 | return new WebResourceRequestAdapter(x5Request); 139 | } 140 | 141 | @Override 142 | public Uri getUrl() { 143 | return mWebResourceRequest.getUrl(); 144 | } 145 | 146 | @Override 147 | public boolean isForMainFrame() { 148 | return mWebResourceRequest.isForMainFrame(); 149 | } 150 | 151 | @Override 152 | public boolean isRedirect() { 153 | return mWebResourceRequest.isRedirect(); 154 | } 155 | 156 | @Override 157 | public boolean hasGesture() { 158 | return mWebResourceRequest.hasGesture(); 159 | } 160 | 161 | @Override 162 | public String getMethod() { 163 | return mWebResourceRequest.getMethod(); 164 | } 165 | 166 | @Override 167 | public Map getRequestHeaders() { 168 | return mWebResourceRequest.getRequestHeaders(); 169 | } 170 | } 171 | 172 | ``` 173 | 174 | 175 | ``` 176 | public class WebResourceResponseAdapter extends com.tencent.smtt.export.external.interfaces.WebResourceResponse { 177 | 178 | private android.webkit.WebResourceResponse mWebResourceResponse; 179 | 180 | private WebResourceResponseAdapter(android.webkit.WebResourceResponse webResourceResponse){ 181 | mWebResourceResponse = webResourceResponse; 182 | } 183 | 184 | public static WebResourceResponseAdapter adapter(android.webkit.WebResourceResponse webResourceResponse){ 185 | if (webResourceResponse == null){ 186 | return null; 187 | } 188 | return new WebResourceResponseAdapter(webResourceResponse); 189 | 190 | } 191 | 192 | @Override 193 | public String getMimeType() { 194 | return mWebResourceResponse.getMimeType(); 195 | } 196 | 197 | @Override 198 | public InputStream getData() { 199 | return mWebResourceResponse.getData(); 200 | } 201 | 202 | @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) 203 | @Override 204 | public int getStatusCode() { 205 | return mWebResourceResponse.getStatusCode(); 206 | } 207 | 208 | @RequiresApi(Build.VERSION_CODES.LOLLIPOP) 209 | @Override 210 | public Map getResponseHeaders() { 211 | return mWebResourceResponse.getResponseHeaders(); 212 | } 213 | 214 | @Override 215 | public String getEncoding() { 216 | return mWebResourceResponse.getEncoding(); 217 | } 218 | 219 | @RequiresApi(Build.VERSION_CODES.LOLLIPOP) 220 | @Override 221 | public String getReasonPhrase() { 222 | return mWebResourceResponse.getReasonPhrase(); 223 | } 224 | } 225 | 226 | 227 | ``` 228 | 229 | 如果你的项目minSdkVersion<21, 在 `mWebView.loadUrl(url)` 之后调用 `WebViewCacheInterceptorInst.getInstance().loadUrl(url,mWebView.getSettings().getUserAgentString())`; 230 | 231 | 232 | --- 233 | 234 | ### 进阶设置 235 | 236 | - 基本设置 237 | 238 | ```Java 239 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 240 | 241 | builder.setCachePath(new File(this.getCacheDir(),"cache_path_name"))//设置缓存路径,默认getCacheDir,名称CacheWebViewCache 242 | .setDynamicCachePath(new File(this.getCacheDir(),"dynamic_webview_cache")) 243 | .setCacheSize(1024*1024*100)//设置缓存大小,默认100M 244 | .setConnectTimeoutSecond(20)//设置http请求链接超时,默认20秒 245 | .setReadTimeoutSecond(20)//设置http请求链接读取超时,默认20秒 246 | .setCacheType(CacheType.NORMAL);//设置缓存为正常模式,默认模式为强制缓存静态资源 247 | 248 | WebViewCacheInterceptorInst.getInstance().init(builder); 249 | ``` 250 | 251 | - 设置缓存后缀 252 | 253 | CacheWebview通过后缀判断来缓存静态文件,可以添加删除 254 | 255 | ```Java 256 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 257 | 258 | CacheExtensionConfig extension = new CacheExtensionConfig(); 259 | extension.addExtension("json").removeExtension("swf");//添加删除缓存后缀 260 | 261 | builder.setCacheExtensionConfig(extension); 262 | 263 | WebViewCacheInterceptorInst.getInstance().init(builder); 264 | ``` 265 | 266 | 默认有以下后缀缓存 267 | 268 | ``` 269 | private static HashSet STATIC = new HashSet() { 270 | { 271 | add("html"); 272 | add("htm"); 273 | add("js"); 274 | add("ico"); 275 | add("css"); 276 | add("png"); 277 | add("jpg"); 278 | add("jpeg"); 279 | add("gif"); 280 | add("bmp"); 281 | add("ttf"); 282 | add("woff"); 283 | add("woff2"); 284 | add("otf"); 285 | add("eot"); 286 | add("svg"); 287 | add("xml"); 288 | add("swf"); 289 | add("txt"); 290 | add("text"); 291 | add("conf"); 292 | add("webp"); 293 | } 294 | }; 295 | 296 | ``` 297 | 298 | 默认有以下后缀不缓存 299 | 300 | ``` 301 | private static HashSet NO_CACH = new HashSet() { 302 | { 303 | add("mp4"); 304 | add("mp3"); 305 | add("ogg"); 306 | add("avi"); 307 | add("wmv"); 308 | add("flv"); 309 | add("rmvb"); 310 | add("3gp"); 311 | } 312 | }; 313 | ``` 314 | 315 | - 设置Assets路径 316 | 317 | CacheWebview可以从Assets路径加载静态资源,只要设置了Assets路径就是开启此功能,默认未开启; 318 | 319 | ``` 320 | 321 | 322 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 323 | //默认精确匹配地址规则 324 | builder.setAssetsDir("static"); 325 | 326 | //后缀匹配规则 327 | //builder.isAssetsSuffixMod(true); 328 | //WebViewCacheInterceptorInst.getInstance().initAssetsData(); //后台线程获取Assets文件资源 329 | 330 | WebViewCacheInterceptorInst.getInstance().init(builder); 331 | 332 | ``` 333 | 334 | builder.setAssetsDir("static")后匹配规则: 335 | 336 | assets 结构如下: 337 | 338 | ![](art/assets.png) 339 | 340 | (1)默认精确匹配规则:那么只有满足这种结构的url:http://xxx.com/scripts/jquery.min.js 都会从assets获取资源 341 | 342 | (2)后缀匹配规则:那么只要满足这种结构的url:http://xxx.com/x/xx/scripts/jquery.min.js 都会从assets获取资源 343 | 344 | - 自定义拦截规则 345 | 346 | ``` 347 | builder.setResourceInterceptor(new ResourceInterceptor() { 348 | @Override 349 | public boolean interceptor(String url) { 350 | return true;//按照默认规则,false 不拦截资源 351 | } 352 | }); 353 | ``` 354 | 355 | 356 | - 获取缓存文件 357 | 358 | ``` 359 | 360 | String url = "http://m.mm131.com/css/at.js"; 361 | InputStream inputStream = WebViewCacheInterceptorInst.getInstance().getCacheFile(url); 362 | if (inputStream!=null){ 363 | 364 | } 365 | 366 | ``` 367 | 368 | - 清除缓存文件 369 | 370 | ``` 371 | WebViewCacheInterceptorInst.getInstance().clearCache(); 372 | ``` 373 | 374 | - 强制缓存失效 375 | 376 | 强制缓存失效后,由WebView正常加载资源 377 | 378 | ``` 379 | WebViewCacheInterceptorInst.getInstance().enableForce(false); 380 | ``` 381 | 382 | - HostnameVerifier设置 383 | 384 | builder.setTrustAllHostname();不安全 385 | 386 | ``` 387 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 388 | builder.setTrustAllHostname();//HostnameVerifier不验证,HostnameVerifier.verify()返回true,默认正常验证 389 | WebViewCacheInterceptorInst.getInstance().init(builder); 390 | 391 | ``` 392 | 393 | - SSLSocketFactory 设置 394 | 395 | ``` 396 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 397 | builder.setSSLSocketFactory(SSLSocketFactory sslSocketFactory, X509TrustManager trustManager);//自定义SSLSocketFactory和X509TrustManager 398 | WebViewCacheInterceptorInst.getInstance().init(builder); 399 | 400 | ``` 401 | 402 | - Debug log 403 | 404 | 默认开启debug log , TAG="CacheWebView",可以关闭log 405 | 406 | ``` 407 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 408 | builder.setDebug(false); 409 | WebViewCacheInterceptorInst.getInstance().init(builder); 410 | 411 | ``` 412 | 413 | 414 | - 非单例模式 415 | 416 | **调用方法和单例一样** 417 | 418 | ``` 419 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 420 | WebViewRequestInterceptor webViewRequestInterceptor = builder.build(); 421 | webViewRequestInterceptor.getCacheFile(""); 422 | ``` 423 | 424 | ## 混淆 425 | 426 | ``` 427 | #CacheWebview 428 | -dontwarn ren.yale.android.cachewebviewlib.** 429 | -keep class ren.yale.android.cachewebviewlib.**{*;} 430 | 431 | #okhttp 432 | -dontwarn okhttp3.** 433 | -keep class okhttp3.**{*;} 434 | 435 | #okio 436 | -dontwarn okio.** 437 | -keep class okio.**{*;} 438 | ``` 439 | 440 | 441 | ## 贡献 442 | 443 | [如何贡献代码](https://github.com/yale8848/CacheWebView/blob/master/CONTRIBUTING.md) 444 | 445 | ### 博客 446 | 447 | [如何让Android WebView访问更快](https://my.oschina.net/yale8848/blog/1544298) 448 | -------------------------------------------------------------------------------- /README_EN.md: -------------------------------------------------------------------------------- 1 | # CacheWebView 2 | 3 | [![](https://img.shields.io/badge/jcenter-2.1.8-519dd9.svg)](https://bintray.com/yale8848/maven/CacheWebView/2.1.8) 4 | 5 | CacheWebView is a custom implement of Android WebView resource interceptor. It beyond system WebView cache space 6 | limit, let cache config more simple ,fast and flexible. Visit website by offline. 7 | 8 | ## Why use CacheWebView? 9 | 10 | - let WebView cache space more bigger 11 | - force cache static, it will more fast 12 | - want to get web page resource in cache, e.g , get pic in cache 13 | 14 | ## Usage 15 | 16 | ### use lib 17 | 18 | ```groovy 19 | implementation 'ren.yale.android:cachewebviewlib:2.2.1' 20 | ``` 21 | 22 | 23 | ### Change code 24 | 25 | Init in Application 26 | 27 | ``` 28 | 29 | WebViewCacheInterceptorInst.getInstance(). 30 | init(new WebViewCacheInterceptor.Builder(this)); 31 | 32 | ``` 33 | 34 | 35 | Add WebView Interceptor 36 | 37 | - If your Android project minSdkVersion>=21 38 | 39 | ``` 40 | mWebView.setWebViewClient(new WebViewClient(){ 41 | 42 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 43 | @Nullable 44 | @Override 45 | public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) { 46 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(request); 47 | } 48 | 49 | @Nullable 50 | @Override 51 | public WebResourceResponse shouldInterceptRequest(WebView view, String url) { 52 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(view,url); 53 | } 54 | }); 55 | 56 | ``` 57 | 58 | - If your Android project minSdkVersion<21 59 | 60 | when call `mWebView.loadUrl(url)` replace by `WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,url)` 61 | 62 | ``` 63 | 64 | mWebView.setWebViewClient(new WebViewClient(){ 65 | 66 | 67 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 68 | @Override 69 | public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) { 70 | WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,request.getUrl().toString()); 71 | return true; 72 | } 73 | 74 | @Override 75 | public boolean shouldOverrideUrlLoading(WebView view, String url) { 76 | WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,url); 77 | return true; 78 | } 79 | 80 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 81 | @Nullable 82 | @Override 83 | public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) { 84 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(view, request); 85 | } 86 | 87 | @Nullable 88 | @Override 89 | public WebResourceResponse shouldInterceptRequest(WebView view, String url) { 90 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(view,url); 91 | } 92 | }); 93 | 94 | ``` 95 | 96 | --- 97 | 98 | ### Setting 99 | 100 | - Basic 101 | 102 | ```Java 103 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 104 | 105 | builder.setCachePath(new File(this.getCacheDir(),"cache_path_name"))//set cache path, default getCacheDir, name CacheWebViewCache 106 | .setDynamicCachePath(new File(this.getCacheDir(),"dynamic_webview_cache")) 107 | .setCacheSize(1024*1024*100)//set cache size, default 100M 108 | .setConnectTimeoutSecond(20)//set http connect timeou,default 20 seconds 109 | .setReadTimeoutSecond(20)//set http read timeout,default 20 seconds 110 | .setCacheType(CacheType.NORMAL);//set cache modal is normal, default is force cache static modal 111 | 112 | WebViewCacheInterceptorInst.getInstance().init(builder); 113 | ``` 114 | 115 | - set cache suffix 116 | 117 | CacheWebview according url suffix to cache, you can add and remove suffix 118 | 119 | ```Java 120 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 121 | 122 | CacheExtensionConfig extension = new CacheExtensionConfig(); 123 | extension.addExtension("json").removeExtension("swf"); 124 | 125 | builder.setCacheExtensionConfig(extension); 126 | 127 | WebViewCacheInterceptorInst.getInstance().initAssetsData();//background thread to get assets files 128 | WebViewCacheInterceptorInst.getInstance().init(builder); 129 | ``` 130 | 131 | default cached suffix 132 | 133 | ``` 134 | private static HashSet STATIC = new HashSet() { 135 | { 136 | add("html"); 137 | add("htm"); 138 | add("js"); 139 | add("ico"); 140 | add("css"); 141 | add("png"); 142 | add("jpg"); 143 | add("jpeg"); 144 | add("gif"); 145 | add("bmp"); 146 | add("ttf"); 147 | add("woff"); 148 | add("woff2"); 149 | add("otf"); 150 | add("eot"); 151 | add("svg"); 152 | add("xml"); 153 | add("swf"); 154 | add("txt"); 155 | add("text"); 156 | add("conf"); 157 | add("webp"); 158 | } 159 | }; 160 | 161 | ``` 162 | 163 | default do not cached suffix 164 | 165 | ``` 166 | private static HashSet NO_CACH = new HashSet() { 167 | { 168 | add("mp4"); 169 | add("mp3"); 170 | add("ogg"); 171 | add("avi"); 172 | add("wmv"); 173 | add("flv"); 174 | add("rmvb"); 175 | add("3gp"); 176 | } 177 | }; 178 | ``` 179 | 180 | - set Assets dir 181 | 182 | CacheWebview can get static from Assets, if you set Assets dir, it will read static from Assets, default is not 183 | 184 | ``` 185 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 186 | 187 | builder.setAssetsDir("static"); 188 | 189 | WebViewCacheInterceptorInst.getInstance().init(builder); 190 | ``` 191 | 192 | builder.setAssetsDir("static") match regular: 193 | 194 | assets struct: 195 | 196 | ![](art/assets.png) 197 | 198 | if match like this url:http://xxxxxx/scripts/jquery.min.js , it will be read static from Assets 199 | 200 | 201 | - custom interceptor role 202 | 203 | ``` 204 | builder.setResourceInterceptor(new ResourceInterceptor() { 205 | @Override 206 | public boolean interceptor(String url) { 207 | return true; 208 | } 209 | }); 210 | ``` 211 | 212 | 213 | - get cache file 214 | 215 | ``` 216 | 217 | String url = "http://m.mm131.com/css/at.js"; 218 | InputStream inputStream = WebViewCacheInterceptorInst.getInstance().getCacheFile(url); 219 | if (inputStream!=null){ 220 | 221 | } 222 | 223 | ``` 224 | 225 | - clear cache file 226 | 227 | ``` 228 | WebViewCacheInterceptorInst.getInstance().clearCache(); 229 | ``` 230 | 231 | - set force cache disable 232 | 233 | after set force cache disable, Webview will load static by itself 234 | 235 | ``` 236 | WebViewCacheInterceptorInst.getInstance().enableForce(false); 237 | ``` 238 | 239 | - None singleton used 240 | 241 | **call method same as singleton** 242 | 243 | ``` 244 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 245 | WebViewRequestInterceptor webViewRequestInterceptor = builder.build(); 246 | webViewRequestInterceptor.getCacheFile(""); 247 | ``` 248 | 249 | ## Proguard 250 | 251 | ``` 252 | #CacheWebview 253 | -dontwarn ren.yale.android.cachewebviewlib.** 254 | -keep class ren.yale.android.cachewebviewlib.**{*;} 255 | 256 | #okhttp 257 | -dontwarn okhttp3.** 258 | -keep class okhttp3.**{*;} 259 | 260 | #okio 261 | -dontwarn okio.** 262 | -keep class okio.**{*;} 263 | ``` 264 | 265 | ## How to contribute 266 | 267 | [Contributing Guide](https://github.com/yale8848/CacheWebView/blob/master/CONTRIBUTING.md) 268 | -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 28 5 | buildToolsVersion "28.0.3" 6 | defaultConfig { 7 | applicationId "ren.yale.android.cachewebview" 8 | minSdkVersion 14 9 | targetSdkVersion 21 10 | versionCode 1 11 | versionName "1.0" 12 | testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" 13 | } 14 | buildTypes { 15 | release { 16 | minifyEnabled true 17 | shrinkResources true 18 | zipAlignEnabled true 19 | debuggable false 20 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 21 | } 22 | } 23 | } 24 | 25 | dependencies { 26 | implementation fileTree(include: ['*.jar'], dir: 'libs') 27 | //implementation 'ren.yale.android:cachewebviewlib:2.1.3' 28 | implementation project(':cachewebviewlib') 29 | implementation 'com.android.support:appcompat-v7:27.1.1' 30 | implementation 'com.android.support.constraint:constraint-layout:1.1.2' 31 | implementation 'com.squareup.okhttp3:okhttp:3.10.0' 32 | } 33 | -------------------------------------------------------------------------------- /app/libs/tbs_sdk_thirdapp_v3.6.0.1315_43612_sharewithdownload_withoutGame_obfs_20180718_102847.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yale8848/CacheWebView/a2f52c7e8732fb0932824fce3bac8044b6960ad0/app/libs/tbs_sdk_thirdapp_v3.6.0.1315_43612_sharewithdownload_withoutGame_obfs_20180718_102847.jar -------------------------------------------------------------------------------- /app/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 C:\YaleSoftFiles\Develop\Android-Studio-SDK/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 | 19 | # Uncomment this to preserve the line number information for 20 | # debugging stack traces. 21 | #-keepattributes SourceFile,LineNumberTable 22 | 23 | # If you keep the line number information, uncomment this to 24 | # hide the original source file name. 25 | #-renamesourcefileattribute SourceFile 26 | #代码混淆压缩比,在0~7之间,默认为5,一般不做修改 27 | -optimizationpasses 5 28 | 29 | #混合时不使用大小写混合,混合后的类名为小写 30 | -dontusemixedcaseclassnames 31 | 32 | #指定不去忽略非公共库的类 33 | -dontskipnonpubliclibraryclasses 34 | 35 | #这句话能够使我们的项目混淆后产生映射文件 36 | #包含有类名->混淆后类名的映射关系 37 | -verbose 38 | 39 | #指定不去忽略非公共库的类 40 | -dontskipnonpubliclibraryclassmembers 41 | 42 | #不做预校验,preverify是proguard的四个步骤之一,Android不需要preverify,去掉这一步能够加快混淆速度。 43 | -dontpreverify 44 | 45 | #保留Annotation不混淆 46 | -keepattributes *Annotation*,InnerClasses 47 | 48 | #避免混淆泛型 49 | -keepattributes Signature 50 | 51 | #抛出异常时保留代码行号 52 | -keepattributes SourceFile,LineNumberTable 53 | 54 | #指定混淆是采用的算法,后面的参数是一个过滤器 55 | #这个过滤器是谷歌推荐的算法,一般不做更改 56 | -optimizations !code/simplification/cast,!field/*,!class/merging/* 57 | 58 | #保留我们使用的四大组件,自定义的Application等等这些类不被混淆 59 | #因为这些子类都有可能被外部调用 60 | -keep public class * extends android.app.Activity 61 | -keep public class * extends android.app.Appliction 62 | -keep public class * extends android.app.Service 63 | -keep public class * extends android.content.BroadcastReceiver 64 | -keep public class * extends android.content.ContentProvider 65 | -keep public class * extends android.app.backup.BackupAgentHelper 66 | -keep public class * extends android.preference.Preference 67 | -keep public class * extends android.view.View 68 | -keep public class com.android.vending.licensing.ILicensingService 69 | 70 | #保留support下的所有类及其内部类 71 | -keep class android.support.** {*;} 72 | 73 | #保留R下面的资源 74 | -keep class **.R$* {*;} 75 | 76 | #保留本地native方法不被混淆 77 | -keepclasseswithmembernames class * { 78 | native ; 79 | } 80 | 81 | #保留在Activity中的方法参数是view的方法, 82 | #这样以来我们在layout中写的onClick就不会被影响 83 | -keepclassmembers class * extends android.app.Activity{ 84 | public void *(android.view.View); 85 | } 86 | 87 | #保留枚举类不被混淆 88 | -keepclassmembers enum * { 89 | public static **[] values(); 90 | public static ** valueOf(java.lang.String); 91 | } 92 | 93 | #保留我们自定义控件(继承自View)不被混淆 94 | -keep public class * extends android.view.View{ 95 | *** get*(); 96 | void set*(***); 97 | public (android.content.Context); 98 | public (android.content.Context, android.util.AttributeSet); 99 | public (android.content.Context, android.util.AttributeSet, int); 100 | } 101 | 102 | #保留Parcelable序列化类不被混淆 103 | -keep class * implements android.os.Parcelable { 104 | public static final android.os.Parcelable$Creator *; 105 | } 106 | 107 | #保留Serializable序列化的类不被混淆 108 | -keepclassmembers class * implements java.io.Serializable { 109 | static final long serialVersionUID; 110 | private static final java.io.ObjectStreamField[] serialPersistentFields; 111 | private void writeObject(java.io.ObjectOutputStream); 112 | private void readObject(java.io.ObjectInputStream); 113 | java.lang.Object writeReplace(); 114 | java.lang.Object readResolve(); 115 | } 116 | 117 | #对于带有回调函数的onXXEvent的,不能被混淆 118 | -keepclassmembers class * { 119 | void *(**On*Event); 120 | } 121 | 122 | 123 | -keep class com.tencent.smtt.export.external.**{ 124 | *; 125 | } 126 | 127 | -keep class com.tencent.tbs.video.interfaces.IUserStateChangedListener { 128 | *; 129 | } 130 | 131 | -keep class com.tencent.smtt.sdk.CacheManager { 132 | public *; 133 | } 134 | 135 | -keep class com.tencent.smtt.sdk.CookieManager { 136 | public *; 137 | } 138 | 139 | -keep class com.tencent.smtt.sdk.WebHistoryItem { 140 | public *; 141 | } 142 | 143 | -keep class com.tencent.smtt.sdk.WebViewDatabase { 144 | public *; 145 | } 146 | 147 | -keep class com.tencent.smtt.sdk.WebBackForwardList { 148 | public *; 149 | } 150 | 151 | -keep public class com.tencent.smtt.sdk.WebView { 152 | public ; 153 | public ; 154 | } 155 | 156 | -keep public class com.tencent.smtt.sdk.WebView$HitTestResult { 157 | public static final ; 158 | public java.lang.String getExtra(); 159 | public int getType(); 160 | } 161 | 162 | -keep public class com.tencent.smtt.sdk.WebView$WebViewTransport { 163 | public ; 164 | } 165 | 166 | -keep public class com.tencent.smtt.sdk.WebView$PictureListener { 167 | public ; 168 | public ; 169 | } 170 | 171 | 172 | -keepattributes InnerClasses 173 | 174 | -keep public enum com.tencent.smtt.sdk.WebSettings$** { 175 | *; 176 | } 177 | 178 | -keep public enum com.tencent.smtt.sdk.QbSdk$** { 179 | *; 180 | } 181 | 182 | -keep public class com.tencent.smtt.sdk.WebSettings { 183 | public *; 184 | } 185 | 186 | 187 | -keepattributes Signature 188 | -keep public class com.tencent.smtt.sdk.ValueCallback { 189 | public ; 190 | public ; 191 | } 192 | 193 | -keep public class com.tencent.smtt.sdk.WebViewClient { 194 | public ; 195 | public ; 196 | } 197 | 198 | -keep public class com.tencent.smtt.sdk.DownloadListener { 199 | public ; 200 | public ; 201 | } 202 | 203 | -keep public class com.tencent.smtt.sdk.WebChromeClient { 204 | public ; 205 | public ; 206 | } 207 | 208 | -keep public class com.tencent.smtt.sdk.WebChromeClient$FileChooserParams { 209 | public ; 210 | public ; 211 | } 212 | 213 | -keep class com.tencent.smtt.sdk.SystemWebChromeClient{ 214 | public *; 215 | } 216 | # 1. extension interfaces should be apparent 217 | -keep public class com.tencent.smtt.export.external.extension.interfaces.* { 218 | public protected *; 219 | } 220 | 221 | # 2. interfaces should be apparent 222 | -keep public class com.tencent.smtt.export.external.interfaces.* { 223 | public protected *; 224 | } 225 | 226 | -keep public class com.tencent.smtt.sdk.WebViewCallbackClient { 227 | public protected *; 228 | } 229 | 230 | -keep public class com.tencent.smtt.sdk.WebStorage$QuotaUpdater { 231 | public ; 232 | public ; 233 | } 234 | 235 | -keep public class com.tencent.smtt.sdk.WebIconDatabase { 236 | public ; 237 | public ; 238 | } 239 | 240 | -keep public class com.tencent.smtt.sdk.WebStorage { 241 | public ; 242 | public ; 243 | } 244 | 245 | -keep public class com.tencent.smtt.sdk.DownloadListener { 246 | public ; 247 | public ; 248 | } 249 | 250 | -keep public class com.tencent.smtt.sdk.QbSdk { 251 | public ; 252 | public ; 253 | } 254 | 255 | -keep public class com.tencent.smtt.sdk.QbSdk$PreInitCallback { 256 | public ; 257 | public ; 258 | } 259 | -keep public class com.tencent.smtt.sdk.CookieSyncManager { 260 | public ; 261 | public ; 262 | } 263 | 264 | -keep public class com.tencent.smtt.sdk.Tbs* { 265 | public ; 266 | public ; 267 | } 268 | 269 | -keep public class com.tencent.smtt.utils.LogFileUtils { 270 | public ; 271 | public ; 272 | } 273 | 274 | -keep public class com.tencent.smtt.utils.TbsLog { 275 | public ; 276 | public ; 277 | } 278 | 279 | -keep public class com.tencent.smtt.utils.TbsLogClient { 280 | public ; 281 | public ; 282 | } 283 | 284 | -keep public class com.tencent.smtt.sdk.CookieSyncManager { 285 | public ; 286 | public ; 287 | } 288 | 289 | # Added for game demos 290 | -keep public class com.tencent.smtt.sdk.TBSGamePlayer { 291 | public ; 292 | public ; 293 | } 294 | 295 | -keep public class com.tencent.smtt.sdk.TBSGamePlayerClient* { 296 | public ; 297 | public ; 298 | } 299 | 300 | -keep public class com.tencent.smtt.sdk.TBSGamePlayerClientExtension { 301 | public ; 302 | public ; 303 | } 304 | 305 | -keep public class com.tencent.smtt.sdk.TBSGamePlayerService* { 306 | public ; 307 | public ; 308 | } 309 | 310 | -keep public class com.tencent.smtt.utils.Apn { 311 | public ; 312 | public ; 313 | } 314 | -keep class com.tencent.smtt.** { 315 | *; 316 | } 317 | # end 318 | 319 | 320 | -keep public class com.tencent.smtt.export.external.extension.proxy.ProxyWebViewClientExtension { 321 | public ; 322 | public ; 323 | } 324 | 325 | -keep class MTT.ThirdAppInfoNew { 326 | *; 327 | } 328 | 329 | -keep class com.tencent.mtt.MttTraceEvent { 330 | *; 331 | } 332 | 333 | # Game related 334 | -keep public class com.tencent.smtt.gamesdk.* { 335 | public protected *; 336 | } 337 | 338 | -keep public class com.tencent.smtt.sdk.TBSGameBooter { 339 | public ; 340 | public ; 341 | } 342 | 343 | -keep public class com.tencent.smtt.sdk.TBSGameBaseActivity { 344 | public protected *; 345 | } 346 | 347 | -keep public class com.tencent.smtt.sdk.TBSGameBaseActivityProxy { 348 | public protected *; 349 | } 350 | 351 | -keep public class com.tencent.smtt.gamesdk.internal.TBSGameServiceClient { 352 | public *; 353 | } 354 | #--------------------------------------------------------------------------- -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /app/src/main/assets/static/scripts/jquery.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * jQuery JavaScript Library v1.4.2 3 | * http://jquery.com/ 4 | * 5 | * Copyright 2010, John Resig 6 | * Dual licensed under the MIT or GPL Version 2 licenses. 7 | * http://jquery.org/license 8 | * 9 | * Includes Sizzle.js 10 | * http://sizzlejs.com/ 11 | * Copyright 2010, The Dojo Foundation 12 | * Released under the MIT, BSD, and GPL Licenses. 13 | * 14 | * Date: Sat Feb 13 22:33:48 2010 -0500 15 | */ 16 | (function(A,w){function ma(){if(!c.isReady){try{s.documentElement.doScroll("left")}catch(a){setTimeout(ma,1);return}c.ready()}}function Qa(a,b){b.src?c.ajax({url:b.src,async:false,dataType:"script"}):c.globalEval(b.text||b.textContent||b.innerHTML||"");b.parentNode&&b.parentNode.removeChild(b)}function X(a,b,d,f,e,j){var i=a.length;if(typeof b==="object"){for(var o in b)X(a,o,b[o],f,e,d);return a}if(d!==w){f=!j&&f&&c.isFunction(d);for(o=0;o)[^>]*$|^#([\w-]+)$/,Ua=/^.[^:#\[\.,]*$/,Va=/\S/, 21 | Wa=/^(\s|\u00A0)+|(\s|\u00A0)+$/g,Xa=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,P=navigator.userAgent,xa=false,Q=[],L,$=Object.prototype.toString,aa=Object.prototype.hasOwnProperty,ba=Array.prototype.push,R=Array.prototype.slice,ya=Array.prototype.indexOf;c.fn=c.prototype={init:function(a,b){var d,f;if(!a)return this;if(a.nodeType){this.context=this[0]=a;this.length=1;return this}if(a==="body"&&!b){this.context=s;this[0]=s.body;this.selector="body";this.length=1;return this}if(typeof a==="string")if((d=Ta.exec(a))&& 22 | (d[1]||!b))if(d[1]){f=b?b.ownerDocument||b:s;if(a=Xa.exec(a))if(c.isPlainObject(b)){a=[s.createElement(a[1])];c.fn.attr.call(a,b,true)}else a=[f.createElement(a[1])];else{a=sa([d[1]],[f]);a=(a.cacheable?a.fragment.cloneNode(true):a.fragment).childNodes}return c.merge(this,a)}else{if(b=s.getElementById(d[2])){if(b.id!==d[2])return T.find(a);this.length=1;this[0]=b}this.context=s;this.selector=a;return this}else if(!b&&/^\w+$/.test(a)){this.selector=a;this.context=s;a=s.getElementsByTagName(a);return c.merge(this, 23 | a)}else return!b||b.jquery?(b||T).find(a):c(b).find(a);else if(c.isFunction(a))return T.ready(a);if(a.selector!==w){this.selector=a.selector;this.context=a.context}return c.makeArray(a,this)},selector:"",jquery:"1.4.2",length:0,size:function(){return this.length},toArray:function(){return R.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this.slice(a)[0]:this[a]},pushStack:function(a,b,d){var f=c();c.isArray(a)?ba.apply(f,a):c.merge(f,a);f.prevObject=this;f.context=this.context;if(b=== 24 | "find")f.selector=this.selector+(this.selector?" ":"")+d;else if(b)f.selector=this.selector+"."+b+"("+d+")";return f},each:function(a,b){return c.each(this,a,b)},ready:function(a){c.bindReady();if(c.isReady)a.call(s,c);else Q&&Q.push(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(R.apply(this,arguments),"slice",R.call(arguments).join(","))},map:function(a){return this.pushStack(c.map(this, 25 | function(b,d){return a.call(b,d,b)}))},end:function(){return this.prevObject||c(null)},push:ba,sort:[].sort,splice:[].splice};c.fn.init.prototype=c.fn;c.extend=c.fn.extend=function(){var a=arguments[0]||{},b=1,d=arguments.length,f=false,e,j,i,o;if(typeof a==="boolean"){f=a;a=arguments[1]||{};b=2}if(typeof a!=="object"&&!c.isFunction(a))a={};if(d===b){a=this;--b}for(;b
a"; 34 | var e=d.getElementsByTagName("*"),j=d.getElementsByTagName("a")[0];if(!(!e||!e.length||!j)){c.support={leadingWhitespace:d.firstChild.nodeType===3,tbody:!d.getElementsByTagName("tbody").length,htmlSerialize:!!d.getElementsByTagName("link").length,style:/red/.test(j.getAttribute("style")),hrefNormalized:j.getAttribute("href")==="/a",opacity:/^0.55$/.test(j.style.opacity),cssFloat:!!j.style.cssFloat,checkOn:d.getElementsByTagName("input")[0].value==="on",optSelected:s.createElement("select").appendChild(s.createElement("option")).selected, 35 | parentNode:d.removeChild(d.appendChild(s.createElement("div"))).parentNode===null,deleteExpando:true,checkClone:false,scriptEval:false,noCloneEvent:true,boxModel:null};b.type="text/javascript";try{b.appendChild(s.createTextNode("window."+f+"=1;"))}catch(i){}a.insertBefore(b,a.firstChild);if(A[f]){c.support.scriptEval=true;delete A[f]}try{delete b.test}catch(o){c.support.deleteExpando=false}a.removeChild(b);if(d.attachEvent&&d.fireEvent){d.attachEvent("onclick",function k(){c.support.noCloneEvent= 36 | false;d.detachEvent("onclick",k)});d.cloneNode(true).fireEvent("onclick")}d=s.createElement("div");d.innerHTML="";a=s.createDocumentFragment();a.appendChild(d.firstChild);c.support.checkClone=a.cloneNode(true).cloneNode(true).lastChild.checked;c(function(){var k=s.createElement("div");k.style.width=k.style.paddingLeft="1px";s.body.appendChild(k);c.boxModel=c.support.boxModel=k.offsetWidth===2;s.body.removeChild(k).style.display="none"});a=function(k){var n= 37 | s.createElement("div");k="on"+k;var r=k in n;if(!r){n.setAttribute(k,"return;");r=typeof n[k]==="function"}return r};c.support.submitBubbles=a("submit");c.support.changeBubbles=a("change");a=b=d=e=j=null}})();c.props={"for":"htmlFor","class":"className",readonly:"readOnly",maxlength:"maxLength",cellspacing:"cellSpacing",rowspan:"rowSpan",colspan:"colSpan",tabindex:"tabIndex",usemap:"useMap",frameborder:"frameBorder"};var G="jQuery"+J(),Ya=0,za={};c.extend({cache:{},expando:G,noData:{embed:true,object:true, 38 | applet:true},data:function(a,b,d){if(!(a.nodeName&&c.noData[a.nodeName.toLowerCase()])){a=a==A?za:a;var f=a[G],e=c.cache;if(!f&&typeof b==="string"&&d===w)return null;f||(f=++Ya);if(typeof b==="object"){a[G]=f;e[f]=c.extend(true,{},b)}else if(!e[f]){a[G]=f;e[f]={}}a=e[f];if(d!==w)a[b]=d;return typeof b==="string"?a[b]:a}},removeData:function(a,b){if(!(a.nodeName&&c.noData[a.nodeName.toLowerCase()])){a=a==A?za:a;var d=a[G],f=c.cache,e=f[d];if(b){if(e){delete e[b];c.isEmptyObject(e)&&c.removeData(a)}}else{if(c.support.deleteExpando)delete a[c.expando]; 39 | else a.removeAttribute&&a.removeAttribute(c.expando);delete f[d]}}}});c.fn.extend({data:function(a,b){if(typeof a==="undefined"&&this.length)return c.data(this[0]);else if(typeof a==="object")return this.each(function(){c.data(this,a)});var d=a.split(".");d[1]=d[1]?"."+d[1]:"";if(b===w){var f=this.triggerHandler("getData"+d[1]+"!",[d[0]]);if(f===w&&this.length)f=c.data(this[0],a);return f===w&&d[1]?this.data(d[0]):f}else return this.trigger("setData"+d[1]+"!",[d[0],b]).each(function(){c.data(this, 40 | a,b)})},removeData:function(a){return this.each(function(){c.removeData(this,a)})}});c.extend({queue:function(a,b,d){if(a){b=(b||"fx")+"queue";var f=c.data(a,b);if(!d)return f||[];if(!f||c.isArray(d))f=c.data(a,b,c.makeArray(d));else f.push(d);return f}},dequeue:function(a,b){b=b||"fx";var d=c.queue(a,b),f=d.shift();if(f==="inprogress")f=d.shift();if(f){b==="fx"&&d.unshift("inprogress");f.call(a,function(){c.dequeue(a,b)})}}});c.fn.extend({queue:function(a,b){if(typeof a!=="string"){b=a;a="fx"}if(b=== 41 | w)return c.queue(this[0],a);return this.each(function(){var d=c.queue(this,a,b);a==="fx"&&d[0]!=="inprogress"&&c.dequeue(this,a)})},dequeue:function(a){return this.each(function(){c.dequeue(this,a)})},delay:function(a,b){a=c.fx?c.fx.speeds[a]||a:a;b=b||"fx";return this.queue(b,function(){var d=this;setTimeout(function(){c.dequeue(d,b)},a)})},clearQueue:function(a){return this.queue(a||"fx",[])}});var Aa=/[\n\t]/g,ca=/\s+/,Za=/\r/g,$a=/href|src|style/,ab=/(button|input)/i,bb=/(button|input|object|select|textarea)/i, 42 | cb=/^(a|area)$/i,Ba=/radio|checkbox/;c.fn.extend({attr:function(a,b){return X(this,a,b,true,c.attr)},removeAttr:function(a){return this.each(function(){c.attr(this,a,"");this.nodeType===1&&this.removeAttribute(a)})},addClass:function(a){if(c.isFunction(a))return this.each(function(n){var r=c(this);r.addClass(a.call(this,n,r.attr("class")))});if(a&&typeof a==="string")for(var b=(a||"").split(ca),d=0,f=this.length;d-1)return true;return false},val:function(a){if(a===w){var b=this[0];if(b){if(c.nodeName(b,"option"))return(b.attributes.value||{}).specified?b.value:b.text;if(c.nodeName(b,"select")){var d=b.selectedIndex,f=[],e=b.options;b=b.type==="select-one";if(d<0)return null;var j=b?d:0;for(d=b?d+1:e.length;j=0;else if(c.nodeName(this,"select")){var u=c.makeArray(r);c("option",this).each(function(){this.selected= 47 | c.inArray(c(this).val(),u)>=0});if(!u.length)this.selectedIndex=-1}else this.value=r}})}});c.extend({attrFn:{val:true,css:true,html:true,text:true,data:true,width:true,height:true,offset:true},attr:function(a,b,d,f){if(!a||a.nodeType===3||a.nodeType===8)return w;if(f&&b in c.attrFn)return c(a)[b](d);f=a.nodeType!==1||!c.isXMLDoc(a);var e=d!==w;b=f&&c.props[b]||b;if(a.nodeType===1){var j=$a.test(b);if(b in a&&f&&!j){if(e){b==="type"&&ab.test(a.nodeName)&&a.parentNode&&c.error("type property can't be changed"); 48 | a[b]=d}if(c.nodeName(a,"form")&&a.getAttributeNode(b))return a.getAttributeNode(b).nodeValue;if(b==="tabIndex")return(b=a.getAttributeNode("tabIndex"))&&b.specified?b.value:bb.test(a.nodeName)||cb.test(a.nodeName)&&a.href?0:w;return a[b]}if(!c.support.style&&f&&b==="style"){if(e)a.style.cssText=""+d;return a.style.cssText}e&&a.setAttribute(b,""+d);a=!c.support.hrefNormalized&&f&&j?a.getAttribute(b,2):a.getAttribute(b);return a===null?w:a}return c.style(a,b,d)}});var O=/\.(.*)$/,db=function(a){return a.replace(/[^\w\s\.\|`]/g, 49 | function(b){return"\\"+b})};c.event={add:function(a,b,d,f){if(!(a.nodeType===3||a.nodeType===8)){if(a.setInterval&&a!==A&&!a.frameElement)a=A;var e,j;if(d.handler){e=d;d=e.handler}if(!d.guid)d.guid=c.guid++;if(j=c.data(a)){var i=j.events=j.events||{},o=j.handle;if(!o)j.handle=o=function(){return typeof c!=="undefined"&&!c.event.triggered?c.event.handle.apply(o.elem,arguments):w};o.elem=a;b=b.split(" ");for(var k,n=0,r;k=b[n++];){j=e?c.extend({},e):{handler:d,data:f};if(k.indexOf(".")>-1){r=k.split("."); 50 | k=r.shift();j.namespace=r.slice(0).sort().join(".")}else{r=[];j.namespace=""}j.type=k;j.guid=d.guid;var u=i[k],z=c.event.special[k]||{};if(!u){u=i[k]=[];if(!z.setup||z.setup.call(a,f,r,o)===false)if(a.addEventListener)a.addEventListener(k,o,false);else a.attachEvent&&a.attachEvent("on"+k,o)}if(z.add){z.add.call(a,j);if(!j.handler.guid)j.handler.guid=d.guid}u.push(j);c.event.global[k]=true}a=null}}},global:{},remove:function(a,b,d,f){if(!(a.nodeType===3||a.nodeType===8)){var e,j=0,i,o,k,n,r,u,z=c.data(a), 51 | C=z&&z.events;if(z&&C){if(b&&b.type){d=b.handler;b=b.type}if(!b||typeof b==="string"&&b.charAt(0)==="."){b=b||"";for(e in C)c.event.remove(a,e+b)}else{for(b=b.split(" ");e=b[j++];){n=e;i=e.indexOf(".")<0;o=[];if(!i){o=e.split(".");e=o.shift();k=new RegExp("(^|\\.)"+c.map(o.slice(0).sort(),db).join("\\.(?:.*\\.)?")+"(\\.|$)")}if(r=C[e])if(d){n=c.event.special[e]||{};for(B=f||0;B=0){a.type= 53 | e=e.slice(0,-1);a.exclusive=true}if(!d){a.stopPropagation();c.event.global[e]&&c.each(c.cache,function(){this.events&&this.events[e]&&c.event.trigger(a,b,this.handle.elem)})}if(!d||d.nodeType===3||d.nodeType===8)return w;a.result=w;a.target=d;b=c.makeArray(b);b.unshift(a)}a.currentTarget=d;(f=c.data(d,"handle"))&&f.apply(d,b);f=d.parentNode||d.ownerDocument;try{if(!(d&&d.nodeName&&c.noData[d.nodeName.toLowerCase()]))if(d["on"+e]&&d["on"+e].apply(d,b)===false)a.result=false}catch(j){}if(!a.isPropagationStopped()&& 54 | f)c.event.trigger(a,b,f,true);else if(!a.isDefaultPrevented()){f=a.target;var i,o=c.nodeName(f,"a")&&e==="click",k=c.event.special[e]||{};if((!k._default||k._default.call(d,a)===false)&&!o&&!(f&&f.nodeName&&c.noData[f.nodeName.toLowerCase()])){try{if(f[e]){if(i=f["on"+e])f["on"+e]=null;c.event.triggered=true;f[e]()}}catch(n){}if(i)f["on"+e]=i;c.event.triggered=false}}},handle:function(a){var b,d,f,e;a=arguments[0]=c.event.fix(a||A.event);a.currentTarget=this;b=a.type.indexOf(".")<0&&!a.exclusive; 55 | if(!b){d=a.type.split(".");a.type=d.shift();f=new RegExp("(^|\\.)"+d.slice(0).sort().join("\\.(?:.*\\.)?")+"(\\.|$)")}e=c.data(this,"events");d=e[a.type];if(e&&d){d=d.slice(0);e=0;for(var j=d.length;e-1?c.map(a.options,function(f){return f.selected}).join("-"):"";else if(a.nodeName.toLowerCase()==="select")d=a.selectedIndex;return d},fa=function(a,b){var d=a.target,f,e;if(!(!da.test(d.nodeName)||d.readOnly)){f=c.data(d,"_change_data");e=Fa(d);if(a.type!=="focusout"||d.type!=="radio")c.data(d,"_change_data", 63 | e);if(!(f===w||e===f))if(f!=null||e){a.type="change";return c.event.trigger(a,b,d)}}};c.event.special.change={filters:{focusout:fa,click:function(a){var b=a.target,d=b.type;if(d==="radio"||d==="checkbox"||b.nodeName.toLowerCase()==="select")return fa.call(this,a)},keydown:function(a){var b=a.target,d=b.type;if(a.keyCode===13&&b.nodeName.toLowerCase()!=="textarea"||a.keyCode===32&&(d==="checkbox"||d==="radio")||d==="select-multiple")return fa.call(this,a)},beforeactivate:function(a){a=a.target;c.data(a, 64 | "_change_data",Fa(a))}},setup:function(){if(this.type==="file")return false;for(var a in ea)c.event.add(this,a+".specialChange",ea[a]);return da.test(this.nodeName)},teardown:function(){c.event.remove(this,".specialChange");return da.test(this.nodeName)}};ea=c.event.special.change.filters}s.addEventListener&&c.each({focus:"focusin",blur:"focusout"},function(a,b){function d(f){f=c.event.fix(f);f.type=b;return c.event.handle.call(this,f)}c.event.special[b]={setup:function(){this.addEventListener(a, 65 | d,true)},teardown:function(){this.removeEventListener(a,d,true)}}});c.each(["bind","one"],function(a,b){c.fn[b]=function(d,f,e){if(typeof d==="object"){for(var j in d)this[b](j,f,d[j],e);return this}if(c.isFunction(f)){e=f;f=w}var i=b==="one"?c.proxy(e,function(k){c(this).unbind(k,i);return e.apply(this,arguments)}):e;if(d==="unload"&&b!=="one")this.one(d,f,e);else{j=0;for(var o=this.length;j0){y=t;break}}t=t[g]}m[q]=y}}}var f=/((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^[\]]*\]|['"][^'"]*['"]|[^[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g, 71 | e=0,j=Object.prototype.toString,i=false,o=true;[0,0].sort(function(){o=false;return 0});var k=function(g,h,l,m){l=l||[];var q=h=h||s;if(h.nodeType!==1&&h.nodeType!==9)return[];if(!g||typeof g!=="string")return l;for(var p=[],v,t,y,S,H=true,M=x(h),I=g;(f.exec(""),v=f.exec(I))!==null;){I=v[3];p.push(v[1]);if(v[2]){S=v[3];break}}if(p.length>1&&r.exec(g))if(p.length===2&&n.relative[p[0]])t=ga(p[0]+p[1],h);else for(t=n.relative[p[0]]?[h]:k(p.shift(),h);p.length;){g=p.shift();if(n.relative[g])g+=p.shift(); 72 | t=ga(g,t)}else{if(!m&&p.length>1&&h.nodeType===9&&!M&&n.match.ID.test(p[0])&&!n.match.ID.test(p[p.length-1])){v=k.find(p.shift(),h,M);h=v.expr?k.filter(v.expr,v.set)[0]:v.set[0]}if(h){v=m?{expr:p.pop(),set:z(m)}:k.find(p.pop(),p.length===1&&(p[0]==="~"||p[0]==="+")&&h.parentNode?h.parentNode:h,M);t=v.expr?k.filter(v.expr,v.set):v.set;if(p.length>0)y=z(t);else H=false;for(;p.length;){var D=p.pop();v=D;if(n.relative[D])v=p.pop();else D="";if(v==null)v=h;n.relative[D](y,v,M)}}else y=[]}y||(y=t);y||k.error(D|| 73 | g);if(j.call(y)==="[object Array]")if(H)if(h&&h.nodeType===1)for(g=0;y[g]!=null;g++){if(y[g]&&(y[g]===true||y[g].nodeType===1&&E(h,y[g])))l.push(t[g])}else for(g=0;y[g]!=null;g++)y[g]&&y[g].nodeType===1&&l.push(t[g]);else l.push.apply(l,y);else z(y,l);if(S){k(S,q,l,m);k.uniqueSort(l)}return l};k.uniqueSort=function(g){if(B){i=o;g.sort(B);if(i)for(var h=1;h":function(g,h){var l=typeof h==="string";if(l&&!/\W/.test(h)){h=h.toLowerCase();for(var m=0,q=g.length;m=0))l||m.push(v);else if(l)h[p]=false;return false},ID:function(g){return g[1].replace(/\\/g,"")},TAG:function(g){return g[1].toLowerCase()}, 80 | CHILD:function(g){if(g[1]==="nth"){var h=/(-?)(\d*)n((?:\+|-)?\d*)/.exec(g[2]==="even"&&"2n"||g[2]==="odd"&&"2n+1"||!/\D/.test(g[2])&&"0n+"+g[2]||g[2]);g[2]=h[1]+(h[2]||1)-0;g[3]=h[3]-0}g[0]=e++;return g},ATTR:function(g,h,l,m,q,p){h=g[1].replace(/\\/g,"");if(!p&&n.attrMap[h])g[1]=n.attrMap[h];if(g[2]==="~=")g[4]=" "+g[4]+" ";return g},PSEUDO:function(g,h,l,m,q){if(g[1]==="not")if((f.exec(g[3])||"").length>1||/^\w/.test(g[3]))g[3]=k(g[3],null,null,h);else{g=k.filter(g[3],h,l,true^q);l||m.push.apply(m, 81 | g);return false}else if(n.match.POS.test(g[0])||n.match.CHILD.test(g[0]))return true;return g},POS:function(g){g.unshift(true);return g}},filters:{enabled:function(g){return g.disabled===false&&g.type!=="hidden"},disabled:function(g){return g.disabled===true},checked:function(g){return g.checked===true},selected:function(g){return g.selected===true},parent:function(g){return!!g.firstChild},empty:function(g){return!g.firstChild},has:function(g,h,l){return!!k(l[3],g).length},header:function(g){return/h\d/i.test(g.nodeName)}, 82 | text:function(g){return"text"===g.type},radio:function(g){return"radio"===g.type},checkbox:function(g){return"checkbox"===g.type},file:function(g){return"file"===g.type},password:function(g){return"password"===g.type},submit:function(g){return"submit"===g.type},image:function(g){return"image"===g.type},reset:function(g){return"reset"===g.type},button:function(g){return"button"===g.type||g.nodeName.toLowerCase()==="button"},input:function(g){return/input|select|textarea|button/i.test(g.nodeName)}}, 83 | setFilters:{first:function(g,h){return h===0},last:function(g,h,l,m){return h===m.length-1},even:function(g,h){return h%2===0},odd:function(g,h){return h%2===1},lt:function(g,h,l){return hl[3]-0},nth:function(g,h,l){return l[3]-0===h},eq:function(g,h,l){return l[3]-0===h}},filter:{PSEUDO:function(g,h,l,m){var q=h[1],p=n.filters[q];if(p)return p(g,l,h,m);else if(q==="contains")return(g.textContent||g.innerText||a([g])||"").indexOf(h[3])>=0;else if(q==="not"){h= 84 | h[3];l=0;for(m=h.length;l=0}},ID:function(g,h){return g.nodeType===1&&g.getAttribute("id")===h},TAG:function(g,h){return h==="*"&&g.nodeType===1||g.nodeName.toLowerCase()===h},CLASS:function(g,h){return(" "+(g.className||g.getAttribute("class"))+" ").indexOf(h)>-1},ATTR:function(g,h){var l=h[1];g=n.attrHandle[l]?n.attrHandle[l](g):g[l]!=null?g[l]:g.getAttribute(l);l=g+"";var m=h[2];h=h[4];return g==null?m==="!=":m=== 86 | "="?l===h:m==="*="?l.indexOf(h)>=0:m==="~="?(" "+l+" ").indexOf(h)>=0:!h?l&&g!==false:m==="!="?l!==h:m==="^="?l.indexOf(h)===0:m==="$="?l.substr(l.length-h.length)===h:m==="|="?l===h||l.substr(0,h.length+1)===h+"-":false},POS:function(g,h,l,m){var q=n.setFilters[h[2]];if(q)return q(g,l,h,m)}}},r=n.match.POS;for(var u in n.match){n.match[u]=new RegExp(n.match[u].source+/(?![^\[]*\])(?![^\(]*\))/.source);n.leftMatch[u]=new RegExp(/(^(?:.|\r|\n)*?)/.source+n.match[u].source.replace(/\\(\d+)/g,function(g, 87 | h){return"\\"+(h-0+1)}))}var z=function(g,h){g=Array.prototype.slice.call(g,0);if(h){h.push.apply(h,g);return h}return g};try{Array.prototype.slice.call(s.documentElement.childNodes,0)}catch(C){z=function(g,h){h=h||[];if(j.call(g)==="[object Array]")Array.prototype.push.apply(h,g);else if(typeof g.length==="number")for(var l=0,m=g.length;l";var l=s.documentElement;l.insertBefore(g,l.firstChild);if(s.getElementById(h)){n.find.ID=function(m,q,p){if(typeof q.getElementById!=="undefined"&&!p)return(q=q.getElementById(m[1]))?q.id===m[1]||typeof q.getAttributeNode!=="undefined"&& 90 | q.getAttributeNode("id").nodeValue===m[1]?[q]:w:[]};n.filter.ID=function(m,q){var p=typeof m.getAttributeNode!=="undefined"&&m.getAttributeNode("id");return m.nodeType===1&&p&&p.nodeValue===q}}l.removeChild(g);l=g=null})();(function(){var g=s.createElement("div");g.appendChild(s.createComment(""));if(g.getElementsByTagName("*").length>0)n.find.TAG=function(h,l){l=l.getElementsByTagName(h[1]);if(h[1]==="*"){h=[];for(var m=0;l[m];m++)l[m].nodeType===1&&h.push(l[m]);l=h}return l};g.innerHTML=""; 91 | if(g.firstChild&&typeof g.firstChild.getAttribute!=="undefined"&&g.firstChild.getAttribute("href")!=="#")n.attrHandle.href=function(h){return h.getAttribute("href",2)};g=null})();s.querySelectorAll&&function(){var g=k,h=s.createElement("div");h.innerHTML="

";if(!(h.querySelectorAll&&h.querySelectorAll(".TEST").length===0)){k=function(m,q,p,v){q=q||s;if(!v&&q.nodeType===9&&!x(q))try{return z(q.querySelectorAll(m),p)}catch(t){}return g(m,q,p,v)};for(var l in g)k[l]=g[l];h=null}}(); 92 | (function(){var g=s.createElement("div");g.innerHTML="
";if(!(!g.getElementsByClassName||g.getElementsByClassName("e").length===0)){g.lastChild.className="e";if(g.getElementsByClassName("e").length!==1){n.order.splice(1,0,"CLASS");n.find.CLASS=function(h,l,m){if(typeof l.getElementsByClassName!=="undefined"&&!m)return l.getElementsByClassName(h[1])};g=null}}})();var E=s.compareDocumentPosition?function(g,h){return!!(g.compareDocumentPosition(h)&16)}: 93 | function(g,h){return g!==h&&(g.contains?g.contains(h):true)},x=function(g){return(g=(g?g.ownerDocument||g:0).documentElement)?g.nodeName!=="HTML":false},ga=function(g,h){var l=[],m="",q;for(h=h.nodeType?[h]:h;q=n.match.PSEUDO.exec(g);){m+=q[0];g=g.replace(n.match.PSEUDO,"")}g=n.relative[g]?g+"*":g;q=0;for(var p=h.length;q=0===d})};c.fn.extend({find:function(a){for(var b=this.pushStack("","find",a),d=0,f=0,e=this.length;f0)for(var j=d;j0},closest:function(a,b){if(c.isArray(a)){var d=[],f=this[0],e,j= 96 | {},i;if(f&&a.length){e=0;for(var o=a.length;e-1:c(f).is(e)){d.push({selector:i,elem:f});delete j[i]}}f=f.parentNode}}return d}var k=c.expr.match.POS.test(a)?c(a,b||this.context):null;return this.map(function(n,r){for(;r&&r.ownerDocument&&r!==b;){if(k?k.index(r)>-1:c(r).is(a))return r;r=r.parentNode}return null})},index:function(a){if(!a||typeof a=== 97 | "string")return c.inArray(this[0],a?c(a):this.parent().children());return c.inArray(a.jquery?a[0]:a,this)},add:function(a,b){a=typeof a==="string"?c(a,b||this.context):c.makeArray(a);b=c.merge(this.get(),a);return this.pushStack(qa(a[0])||qa(b[0])?b:c.unique(b))},andSelf:function(){return this.add(this.prevObject)}});c.each({parent:function(a){return(a=a.parentNode)&&a.nodeType!==11?a:null},parents:function(a){return c.dir(a,"parentNode")},parentsUntil:function(a,b,d){return c.dir(a,"parentNode", 98 | d)},next:function(a){return c.nth(a,2,"nextSibling")},prev:function(a){return c.nth(a,2,"previousSibling")},nextAll:function(a){return c.dir(a,"nextSibling")},prevAll:function(a){return c.dir(a,"previousSibling")},nextUntil:function(a,b,d){return c.dir(a,"nextSibling",d)},prevUntil:function(a,b,d){return c.dir(a,"previousSibling",d)},siblings:function(a){return c.sibling(a.parentNode.firstChild,a)},children:function(a){return c.sibling(a.firstChild)},contents:function(a){return c.nodeName(a,"iframe")? 99 | a.contentDocument||a.contentWindow.document:c.makeArray(a.childNodes)}},function(a,b){c.fn[a]=function(d,f){var e=c.map(this,b,d);eb.test(a)||(f=d);if(f&&typeof f==="string")e=c.filter(f,e);e=this.length>1?c.unique(e):e;if((this.length>1||gb.test(f))&&fb.test(a))e=e.reverse();return this.pushStack(e,a,R.call(arguments).join(","))}});c.extend({filter:function(a,b,d){if(d)a=":not("+a+")";return c.find.matches(a,b)},dir:function(a,b,d){var f=[];for(a=a[b];a&&a.nodeType!==9&&(d===w||a.nodeType!==1||!c(a).is(d));){a.nodeType=== 100 | 1&&f.push(a);a=a[b]}return f},nth:function(a,b,d){b=b||1;for(var f=0;a;a=a[d])if(a.nodeType===1&&++f===b)break;return a},sibling:function(a,b){for(var d=[];a;a=a.nextSibling)a.nodeType===1&&a!==b&&d.push(a);return d}});var Ja=/ jQuery\d+="(?:\d+|null)"/g,V=/^\s+/,Ka=/(<([\w:]+)[^>]*?)\/>/g,hb=/^(?:area|br|col|embed|hr|img|input|link|meta|param)$/i,La=/<([\w:]+)/,ib=/"},F={option:[1,""],legend:[1,"
","
"],thead:[1,"","
"],tr:[2,"","
"],td:[3,"","
"],col:[2,"","
"],area:[1,"",""],_default:[0,"",""]};F.optgroup=F.option;F.tbody=F.tfoot=F.colgroup=F.caption=F.thead;F.th=F.td;if(!c.support.htmlSerialize)F._default=[1,"div
","
"];c.fn.extend({text:function(a){if(c.isFunction(a))return this.each(function(b){var d= 102 | c(this);d.text(a.call(this,b,d.text()))});if(typeof a!=="object"&&a!==w)return this.empty().append((this[0]&&this[0].ownerDocument||s).createTextNode(a));return c.text(this)},wrapAll:function(a){if(c.isFunction(a))return this.each(function(d){c(this).wrapAll(a.call(this,d))});if(this[0]){var b=c(a,this[0].ownerDocument).eq(0).clone(true);this[0].parentNode&&b.insertBefore(this[0]);b.map(function(){for(var d=this;d.firstChild&&d.firstChild.nodeType===1;)d=d.firstChild;return d}).append(this)}return this}, 103 | wrapInner:function(a){if(c.isFunction(a))return this.each(function(b){c(this).wrapInner(a.call(this,b))});return this.each(function(){var b=c(this),d=b.contents();d.length?d.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){c(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){c.nodeName(this,"body")||c(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,true,function(a){this.nodeType===1&&this.appendChild(a)})}, 104 | prepend:function(){return this.domManip(arguments,true,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,false,function(b){this.parentNode.insertBefore(b,this)});else if(arguments.length){var a=c(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,false,function(b){this.parentNode.insertBefore(b, 105 | this.nextSibling)});else if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,c(arguments[0]).toArray());return a}},remove:function(a,b){for(var d=0,f;(f=this[d])!=null;d++)if(!a||c.filter(a,[f]).length){if(!b&&f.nodeType===1){c.cleanData(f.getElementsByTagName("*"));c.cleanData([f])}f.parentNode&&f.parentNode.removeChild(f)}return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++)for(b.nodeType===1&&c.cleanData(b.getElementsByTagName("*"));b.firstChild;)b.removeChild(b.firstChild); 106 | return this},clone:function(a){var b=this.map(function(){if(!c.support.noCloneEvent&&!c.isXMLDoc(this)){var d=this.outerHTML,f=this.ownerDocument;if(!d){d=f.createElement("div");d.appendChild(this.cloneNode(true));d=d.innerHTML}return c.clean([d.replace(Ja,"").replace(/=([^="'>\s]+\/)>/g,'="$1">').replace(V,"")],f)[0]}else return this.cloneNode(true)});if(a===true){ra(this,b);ra(this.find("*"),b.find("*"))}return b},html:function(a){if(a===w)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(Ja, 107 | ""):null;else if(typeof a==="string"&&!ta.test(a)&&(c.support.leadingWhitespace||!V.test(a))&&!F[(La.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Ka,Ma);try{for(var b=0,d=this.length;b0||e.cacheable||this.length>1?k.cloneNode(true):k)}o.length&&c.each(o,Qa)}return this}});c.fragments={};c.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){c.fn[a]=function(d){var f=[];d=c(d);var e=this.length===1&&this[0].parentNode;if(e&&e.nodeType===11&&e.childNodes.length===1&&d.length===1){d[b](this[0]); 111 | return this}else{e=0;for(var j=d.length;e0?this.clone(true):this).get();c.fn[b].apply(c(d[e]),i);f=f.concat(i)}return this.pushStack(f,a,d.selector)}}});c.extend({clean:function(a,b,d,f){b=b||s;if(typeof b.createElement==="undefined")b=b.ownerDocument||b[0]&&b[0].ownerDocument||s;for(var e=[],j=0,i;(i=a[j])!=null;j++){if(typeof i==="number")i+="";if(i){if(typeof i==="string"&&!jb.test(i))i=b.createTextNode(i);else if(typeof i==="string"){i=i.replace(Ka,Ma);var o=(La.exec(i)||["", 112 | ""])[1].toLowerCase(),k=F[o]||F._default,n=k[0],r=b.createElement("div");for(r.innerHTML=k[1]+i+k[2];n--;)r=r.lastChild;if(!c.support.tbody){n=ib.test(i);o=o==="table"&&!n?r.firstChild&&r.firstChild.childNodes:k[1]===""&&!n?r.childNodes:[];for(k=o.length-1;k>=0;--k)c.nodeName(o[k],"tbody")&&!o[k].childNodes.length&&o[k].parentNode.removeChild(o[k])}!c.support.leadingWhitespace&&V.test(i)&&r.insertBefore(b.createTextNode(V.exec(i)[0]),r.firstChild);i=r.childNodes}if(i.nodeType)e.push(i);else e= 113 | c.merge(e,i)}}if(d)for(j=0;e[j];j++)if(f&&c.nodeName(e[j],"script")&&(!e[j].type||e[j].type.toLowerCase()==="text/javascript"))f.push(e[j].parentNode?e[j].parentNode.removeChild(e[j]):e[j]);else{e[j].nodeType===1&&e.splice.apply(e,[j+1,0].concat(c.makeArray(e[j].getElementsByTagName("script"))));d.appendChild(e[j])}return e},cleanData:function(a){for(var b,d,f=c.cache,e=c.event.special,j=c.support.deleteExpando,i=0,o;(o=a[i])!=null;i++)if(d=o[c.expando]){b=f[d];if(b.events)for(var k in b.events)e[k]? 114 | c.event.remove(o,k):Ca(o,k,b.handle);if(j)delete o[c.expando];else o.removeAttribute&&o.removeAttribute(c.expando);delete f[d]}}});var kb=/z-?index|font-?weight|opacity|zoom|line-?height/i,Na=/alpha\([^)]*\)/,Oa=/opacity=([^)]*)/,ha=/float/i,ia=/-([a-z])/ig,lb=/([A-Z])/g,mb=/^-?\d+(?:px)?$/i,nb=/^-?\d/,ob={position:"absolute",visibility:"hidden",display:"block"},pb=["Left","Right"],qb=["Top","Bottom"],rb=s.defaultView&&s.defaultView.getComputedStyle,Pa=c.support.cssFloat?"cssFloat":"styleFloat",ja= 115 | function(a,b){return b.toUpperCase()};c.fn.css=function(a,b){return X(this,a,b,true,function(d,f,e){if(e===w)return c.curCSS(d,f);if(typeof e==="number"&&!kb.test(f))e+="px";c.style(d,f,e)})};c.extend({style:function(a,b,d){if(!a||a.nodeType===3||a.nodeType===8)return w;if((b==="width"||b==="height")&&parseFloat(d)<0)d=w;var f=a.style||a,e=d!==w;if(!c.support.opacity&&b==="opacity"){if(e){f.zoom=1;b=parseInt(d,10)+""==="NaN"?"":"alpha(opacity="+d*100+")";a=f.filter||c.curCSS(a,"filter")||"";f.filter= 116 | Na.test(a)?a.replace(Na,b):b}return f.filter&&f.filter.indexOf("opacity=")>=0?parseFloat(Oa.exec(f.filter)[1])/100+"":""}if(ha.test(b))b=Pa;b=b.replace(ia,ja);if(e)f[b]=d;return f[b]},css:function(a,b,d,f){if(b==="width"||b==="height"){var e,j=b==="width"?pb:qb;function i(){e=b==="width"?a.offsetWidth:a.offsetHeight;f!=="border"&&c.each(j,function(){f||(e-=parseFloat(c.curCSS(a,"padding"+this,true))||0);if(f==="margin")e+=parseFloat(c.curCSS(a,"margin"+this,true))||0;else e-=parseFloat(c.curCSS(a, 117 | "border"+this+"Width",true))||0})}a.offsetWidth!==0?i():c.swap(a,ob,i);return Math.max(0,Math.round(e))}return c.curCSS(a,b,d)},curCSS:function(a,b,d){var f,e=a.style;if(!c.support.opacity&&b==="opacity"&&a.currentStyle){f=Oa.test(a.currentStyle.filter||"")?parseFloat(RegExp.$1)/100+"":"";return f===""?"1":f}if(ha.test(b))b=Pa;if(!d&&e&&e[b])f=e[b];else if(rb){if(ha.test(b))b="float";b=b.replace(lb,"-$1").toLowerCase();e=a.ownerDocument.defaultView;if(!e)return null;if(a=e.getComputedStyle(a,null))f= 118 | a.getPropertyValue(b);if(b==="opacity"&&f==="")f="1"}else if(a.currentStyle){d=b.replace(ia,ja);f=a.currentStyle[b]||a.currentStyle[d];if(!mb.test(f)&&nb.test(f)){b=e.left;var j=a.runtimeStyle.left;a.runtimeStyle.left=a.currentStyle.left;e.left=d==="fontSize"?"1em":f||0;f=e.pixelLeft+"px";e.left=b;a.runtimeStyle.left=j}}return f},swap:function(a,b,d){var f={};for(var e in b){f[e]=a.style[e];a.style[e]=b[e]}d.call(a);for(e in b)a.style[e]=f[e]}});if(c.expr&&c.expr.filters){c.expr.filters.hidden=function(a){var b= 119 | a.offsetWidth,d=a.offsetHeight,f=a.nodeName.toLowerCase()==="tr";return b===0&&d===0&&!f?true:b>0&&d>0&&!f?false:c.curCSS(a,"display")==="none"};c.expr.filters.visible=function(a){return!c.expr.filters.hidden(a)}}var sb=J(),tb=//gi,ub=/select|textarea/i,vb=/color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week/i,N=/=\?(&|$)/,ka=/\?/,wb=/(\?|&)_=.*?(&|$)/,xb=/^(\w+:)?\/\/([^\/?#]+)/,yb=/%20/g,zb=c.fn.load;c.fn.extend({load:function(a,b,d){if(typeof a!== 120 | "string")return zb.call(this,a);else if(!this.length)return this;var f=a.indexOf(" ");if(f>=0){var e=a.slice(f,a.length);a=a.slice(0,f)}f="GET";if(b)if(c.isFunction(b)){d=b;b=null}else if(typeof b==="object"){b=c.param(b,c.ajaxSettings.traditional);f="POST"}var j=this;c.ajax({url:a,type:f,dataType:"html",data:b,complete:function(i,o){if(o==="success"||o==="notmodified")j.html(e?c("
").append(i.responseText.replace(tb,"")).find(e):i.responseText);d&&j.each(d,[i.responseText,o,i])}});return this}, 121 | serialize:function(){return c.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?c.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||ub.test(this.nodeName)||vb.test(this.type))}).map(function(a,b){a=c(this).val();return a==null?null:c.isArray(a)?c.map(a,function(d){return{name:b.name,value:d}}):{name:b.name,value:a}}).get()}});c.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "), 122 | function(a,b){c.fn[b]=function(d){return this.bind(b,d)}});c.extend({get:function(a,b,d,f){if(c.isFunction(b)){f=f||d;d=b;b=null}return c.ajax({type:"GET",url:a,data:b,success:d,dataType:f})},getScript:function(a,b){return c.get(a,null,b,"script")},getJSON:function(a,b,d){return c.get(a,b,d,"json")},post:function(a,b,d,f){if(c.isFunction(b)){f=f||d;d=b;b={}}return c.ajax({type:"POST",url:a,data:b,success:d,dataType:f})},ajaxSetup:function(a){c.extend(c.ajaxSettings,a)},ajaxSettings:{url:location.href, 123 | global:true,type:"GET",contentType:"application/x-www-form-urlencoded",processData:true,async:true,xhr:A.XMLHttpRequest&&(A.location.protocol!=="file:"||!A.ActiveXObject)?function(){return new A.XMLHttpRequest}:function(){try{return new A.ActiveXObject("Microsoft.XMLHTTP")}catch(a){}},accepts:{xml:"application/xml, text/xml",html:"text/html",script:"text/javascript, application/javascript",json:"application/json, text/javascript",text:"text/plain",_default:"*/*"}},lastModified:{},etag:{},ajax:function(a){function b(){e.success&& 124 | e.success.call(k,o,i,x);e.global&&f("ajaxSuccess",[x,e])}function d(){e.complete&&e.complete.call(k,x,i);e.global&&f("ajaxComplete",[x,e]);e.global&&!--c.active&&c.event.trigger("ajaxStop")}function f(q,p){(e.context?c(e.context):c.event).trigger(q,p)}var e=c.extend(true,{},c.ajaxSettings,a),j,i,o,k=a&&a.context||e,n=e.type.toUpperCase();if(e.data&&e.processData&&typeof e.data!=="string")e.data=c.param(e.data,e.traditional);if(e.dataType==="jsonp"){if(n==="GET")N.test(e.url)||(e.url+=(ka.test(e.url)? 125 | "&":"?")+(e.jsonp||"callback")+"=?");else if(!e.data||!N.test(e.data))e.data=(e.data?e.data+"&":"")+(e.jsonp||"callback")+"=?";e.dataType="json"}if(e.dataType==="json"&&(e.data&&N.test(e.data)||N.test(e.url))){j=e.jsonpCallback||"jsonp"+sb++;if(e.data)e.data=(e.data+"").replace(N,"="+j+"$1");e.url=e.url.replace(N,"="+j+"$1");e.dataType="script";A[j]=A[j]||function(q){o=q;b();d();A[j]=w;try{delete A[j]}catch(p){}z&&z.removeChild(C)}}if(e.dataType==="script"&&e.cache===null)e.cache=false;if(e.cache=== 126 | false&&n==="GET"){var r=J(),u=e.url.replace(wb,"$1_="+r+"$2");e.url=u+(u===e.url?(ka.test(e.url)?"&":"?")+"_="+r:"")}if(e.data&&n==="GET")e.url+=(ka.test(e.url)?"&":"?")+e.data;e.global&&!c.active++&&c.event.trigger("ajaxStart");r=(r=xb.exec(e.url))&&(r[1]&&r[1]!==location.protocol||r[2]!==location.host);if(e.dataType==="script"&&n==="GET"&&r){var z=s.getElementsByTagName("head")[0]||s.documentElement,C=s.createElement("script");C.src=e.url;if(e.scriptCharset)C.charset=e.scriptCharset;if(!j){var B= 127 | false;C.onload=C.onreadystatechange=function(){if(!B&&(!this.readyState||this.readyState==="loaded"||this.readyState==="complete")){B=true;b();d();C.onload=C.onreadystatechange=null;z&&C.parentNode&&z.removeChild(C)}}}z.insertBefore(C,z.firstChild);return w}var E=false,x=e.xhr();if(x){e.username?x.open(n,e.url,e.async,e.username,e.password):x.open(n,e.url,e.async);try{if(e.data||a&&a.contentType)x.setRequestHeader("Content-Type",e.contentType);if(e.ifModified){c.lastModified[e.url]&&x.setRequestHeader("If-Modified-Since", 128 | c.lastModified[e.url]);c.etag[e.url]&&x.setRequestHeader("If-None-Match",c.etag[e.url])}r||x.setRequestHeader("X-Requested-With","XMLHttpRequest");x.setRequestHeader("Accept",e.dataType&&e.accepts[e.dataType]?e.accepts[e.dataType]+", */*":e.accepts._default)}catch(ga){}if(e.beforeSend&&e.beforeSend.call(k,x,e)===false){e.global&&!--c.active&&c.event.trigger("ajaxStop");x.abort();return false}e.global&&f("ajaxSend",[x,e]);var g=x.onreadystatechange=function(q){if(!x||x.readyState===0||q==="abort"){E|| 129 | d();E=true;if(x)x.onreadystatechange=c.noop}else if(!E&&x&&(x.readyState===4||q==="timeout")){E=true;x.onreadystatechange=c.noop;i=q==="timeout"?"timeout":!c.httpSuccess(x)?"error":e.ifModified&&c.httpNotModified(x,e.url)?"notmodified":"success";var p;if(i==="success")try{o=c.httpData(x,e.dataType,e)}catch(v){i="parsererror";p=v}if(i==="success"||i==="notmodified")j||b();else c.handleError(e,x,i,p);d();q==="timeout"&&x.abort();if(e.async)x=null}};try{var h=x.abort;x.abort=function(){x&&h.call(x); 130 | g("abort")}}catch(l){}e.async&&e.timeout>0&&setTimeout(function(){x&&!E&&g("timeout")},e.timeout);try{x.send(n==="POST"||n==="PUT"||n==="DELETE"?e.data:null)}catch(m){c.handleError(e,x,null,m);d()}e.async||g();return x}},handleError:function(a,b,d,f){if(a.error)a.error.call(a.context||a,b,d,f);if(a.global)(a.context?c(a.context):c.event).trigger("ajaxError",[b,a,f])},active:0,httpSuccess:function(a){try{return!a.status&&location.protocol==="file:"||a.status>=200&&a.status<300||a.status===304||a.status=== 131 | 1223||a.status===0}catch(b){}return false},httpNotModified:function(a,b){var d=a.getResponseHeader("Last-Modified"),f=a.getResponseHeader("Etag");if(d)c.lastModified[b]=d;if(f)c.etag[b]=f;return a.status===304||a.status===0},httpData:function(a,b,d){var f=a.getResponseHeader("content-type")||"",e=b==="xml"||!b&&f.indexOf("xml")>=0;a=e?a.responseXML:a.responseText;e&&a.documentElement.nodeName==="parsererror"&&c.error("parsererror");if(d&&d.dataFilter)a=d.dataFilter(a,b);if(typeof a==="string")if(b=== 132 | "json"||!b&&f.indexOf("json")>=0)a=c.parseJSON(a);else if(b==="script"||!b&&f.indexOf("javascript")>=0)c.globalEval(a);return a},param:function(a,b){function d(i,o){if(c.isArray(o))c.each(o,function(k,n){b||/\[\]$/.test(i)?f(i,n):d(i+"["+(typeof n==="object"||c.isArray(n)?k:"")+"]",n)});else!b&&o!=null&&typeof o==="object"?c.each(o,function(k,n){d(i+"["+k+"]",n)}):f(i,o)}function f(i,o){o=c.isFunction(o)?o():o;e[e.length]=encodeURIComponent(i)+"="+encodeURIComponent(o)}var e=[];if(b===w)b=c.ajaxSettings.traditional; 133 | if(c.isArray(a)||a.jquery)c.each(a,function(){f(this.name,this.value)});else for(var j in a)d(j,a[j]);return e.join("&").replace(yb,"+")}});var la={},Ab=/toggle|show|hide/,Bb=/^([+-]=)?([\d+-.]+)(.*)$/,W,va=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]];c.fn.extend({show:function(a,b){if(a||a===0)return this.animate(K("show",3),a,b);else{a=0;for(b=this.length;a").appendTo("body");f=e.css("display");if(f==="none")f="block";e.remove();la[d]=f}c.data(this[a],"olddisplay",f)}}a=0;for(b=this.length;a=0;f--)if(d[f].elem===this){b&&d[f](true);d.splice(f,1)}});b||this.dequeue();return this}});c.each({slideDown:K("show",1),slideUp:K("hide",1),slideToggle:K("toggle",1),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"}},function(a,b){c.fn[a]=function(d,f){return this.animate(b,d,f)}});c.extend({speed:function(a,b,d){var f=a&&typeof a==="object"?a:{complete:d||!d&&b||c.isFunction(a)&&a,duration:a,easing:d&&b||b&&!c.isFunction(b)&&b};f.duration=c.fx.off?0:typeof f.duration=== 139 | "number"?f.duration:c.fx.speeds[f.duration]||c.fx.speeds._default;f.old=f.complete;f.complete=function(){f.queue!==false&&c(this).dequeue();c.isFunction(f.old)&&f.old.call(this)};return f},easing:{linear:function(a,b,d,f){return d+f*a},swing:function(a,b,d,f){return(-Math.cos(a*Math.PI)/2+0.5)*f+d}},timers:[],fx:function(a,b,d){this.options=b;this.elem=a;this.prop=d;if(!b.orig)b.orig={}}});c.fx.prototype={update:function(){this.options.step&&this.options.step.call(this.elem,this.now,this);(c.fx.step[this.prop]|| 140 | c.fx.step._default)(this);if((this.prop==="height"||this.prop==="width")&&this.elem.style)this.elem.style.display="block"},cur:function(a){if(this.elem[this.prop]!=null&&(!this.elem.style||this.elem.style[this.prop]==null))return this.elem[this.prop];return(a=parseFloat(c.css(this.elem,this.prop,a)))&&a>-10000?a:parseFloat(c.curCSS(this.elem,this.prop))||0},custom:function(a,b,d){function f(j){return e.step(j)}this.startTime=J();this.start=a;this.end=b;this.unit=d||this.unit||"px";this.now=this.start; 141 | this.pos=this.state=0;var e=this;f.elem=this.elem;if(f()&&c.timers.push(f)&&!W)W=setInterval(c.fx.tick,13)},show:function(){this.options.orig[this.prop]=c.style(this.elem,this.prop);this.options.show=true;this.custom(this.prop==="width"||this.prop==="height"?1:0,this.cur());c(this.elem).show()},hide:function(){this.options.orig[this.prop]=c.style(this.elem,this.prop);this.options.hide=true;this.custom(this.cur(),0)},step:function(a){var b=J(),d=true;if(a||b>=this.options.duration+this.startTime){this.now= 142 | this.end;this.pos=this.state=1;this.update();this.options.curAnim[this.prop]=true;for(var f in this.options.curAnim)if(this.options.curAnim[f]!==true)d=false;if(d){if(this.options.display!=null){this.elem.style.overflow=this.options.overflow;a=c.data(this.elem,"olddisplay");this.elem.style.display=a?a:this.options.display;if(c.css(this.elem,"display")==="none")this.elem.style.display="block"}this.options.hide&&c(this.elem).hide();if(this.options.hide||this.options.show)for(var e in this.options.curAnim)c.style(this.elem, 143 | e,this.options.orig[e]);this.options.complete.call(this.elem)}return false}else{e=b-this.startTime;this.state=e/this.options.duration;a=this.options.easing||(c.easing.swing?"swing":"linear");this.pos=c.easing[this.options.specialEasing&&this.options.specialEasing[this.prop]||a](this.state,e,0,1,this.options.duration);this.now=this.start+(this.end-this.start)*this.pos;this.update()}return true}};c.extend(c.fx,{tick:function(){for(var a=c.timers,b=0;b
"; 149 | a.insertBefore(b,a.firstChild);d=b.firstChild;f=d.firstChild;e=d.nextSibling.firstChild.firstChild;this.doesNotAddBorder=f.offsetTop!==5;this.doesAddBorderForTableAndCells=e.offsetTop===5;f.style.position="fixed";f.style.top="20px";this.supportsFixedPosition=f.offsetTop===20||f.offsetTop===15;f.style.position=f.style.top="";d.style.overflow="hidden";d.style.position="relative";this.subtractsBorderForOverflowNotVisible=f.offsetTop===-5;this.doesNotIncludeMarginInBodyOffset=a.offsetTop!==j;a.removeChild(b); 150 | c.offset.initialize=c.noop},bodyOffset:function(a){var b=a.offsetTop,d=a.offsetLeft;c.offset.initialize();if(c.offset.doesNotIncludeMarginInBodyOffset){b+=parseFloat(c.curCSS(a,"marginTop",true))||0;d+=parseFloat(c.curCSS(a,"marginLeft",true))||0}return{top:b,left:d}},setOffset:function(a,b,d){if(/static/.test(c.curCSS(a,"position")))a.style.position="relative";var f=c(a),e=f.offset(),j=parseInt(c.curCSS(a,"top",true),10)||0,i=parseInt(c.curCSS(a,"left",true),10)||0;if(c.isFunction(b))b=b.call(a, 151 | d,e);d={top:b.top-e.top+j,left:b.left-e.left+i};"using"in b?b.using.call(a,d):f.css(d)}};c.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),d=this.offset(),f=/^body|html$/i.test(b[0].nodeName)?{top:0,left:0}:b.offset();d.top-=parseFloat(c.curCSS(a,"marginTop",true))||0;d.left-=parseFloat(c.curCSS(a,"marginLeft",true))||0;f.top+=parseFloat(c.curCSS(b[0],"borderTopWidth",true))||0;f.left+=parseFloat(c.curCSS(b[0],"borderLeftWidth",true))||0;return{top:d.top- 152 | f.top,left:d.left-f.left}},offsetParent:function(){return this.map(function(){for(var a=this.offsetParent||s.body;a&&!/^body|html$/i.test(a.nodeName)&&c.css(a,"position")==="static";)a=a.offsetParent;return a})}});c.each(["Left","Top"],function(a,b){var d="scroll"+b;c.fn[d]=function(f){var e=this[0],j;if(!e)return null;if(f!==w)return this.each(function(){if(j=wa(this))j.scrollTo(!a?f:c(j).scrollLeft(),a?f:c(j).scrollTop());else this[d]=f});else return(j=wa(e))?"pageXOffset"in j?j[a?"pageYOffset": 153 | "pageXOffset"]:c.support.boxModel&&j.document.documentElement[d]||j.document.body[d]:e[d]}});c.each(["Height","Width"],function(a,b){var d=b.toLowerCase();c.fn["inner"+b]=function(){return this[0]?c.css(this[0],d,false,"padding"):null};c.fn["outer"+b]=function(f){return this[0]?c.css(this[0],d,false,f?"margin":"border"):null};c.fn[d]=function(f){var e=this[0];if(!e)return f==null?null:this;if(c.isFunction(f))return this.each(function(j){var i=c(this);i[d](f.call(this,j,i[d]()))});return"scrollTo"in 154 | e&&e.document?e.document.compatMode==="CSS1Compat"&&e.document.documentElement["client"+b]||e.document.body["client"+b]:e.nodeType===9?Math.max(e.documentElement["client"+b],e.body["scroll"+b],e.documentElement["scroll"+b],e.body["offset"+b],e.documentElement["offset"+b]):f===w?c.css(e,d):this.css(d,typeof f==="string"?f:f+"px")}});A.jQuery=A.$=c})(window); -------------------------------------------------------------------------------- /app/src/main/java/ren/yale/android/cachewebview/App.java: -------------------------------------------------------------------------------- 1 | package ren.yale.android.cachewebview; 2 | 3 | import android.app.Application; 4 | import android.util.Log; 5 | 6 | import com.tencent.smtt.sdk.QbSdk; 7 | 8 | import java.io.File; 9 | 10 | import ren.yale.android.cachewebviewlib.ResourceInterceptor; 11 | import ren.yale.android.cachewebviewlib.WebViewCacheInterceptor; 12 | import ren.yale.android.cachewebviewlib.WebViewCacheInterceptorInst; 13 | import ren.yale.android.cachewebviewlib.config.CacheExtensionConfig; 14 | 15 | /** 16 | * Created by yale on 2017/9/27. 17 | */ 18 | 19 | public class App extends Application { 20 | private static final String CACHE_NAME = "cache_path"; 21 | 22 | @Override 23 | public void onCreate() { 24 | super.onCreate(); 25 | 26 | WebViewCacheInterceptor.Builder builder = new WebViewCacheInterceptor.Builder(this); 27 | 28 | //设置okhttp缓存路径,默认getCacheDir,名称CacheWebViewCache 29 | builder.setCachePath(new File(this.getCacheDir(),"cache_path_name")) 30 | .setDynamicCachePath(new File(this.getCacheDir(),"dynamic_webview_cache")) 31 | .setCacheSize(1024*1024*100)//设置缓存大小,默认100M 32 | .setConnectTimeoutSecond(20)//设置http请求链接超时,默认20秒 33 | .setReadTimeoutSecond(20);//设置http请求链接读取超时,默认20秒 34 | 35 | 36 | CacheExtensionConfig extension = new CacheExtensionConfig(); 37 | extension.addExtension("json").removeExtension("swf"); 38 | 39 | builder.setCacheExtensionConfig(extension); 40 | //builder.setAssetsDir("static"); 41 | //builder.isAssetsSuffixMod(true); 42 | builder.setDebug(true); 43 | 44 | builder.setResourceInterceptor(new ResourceInterceptor() { 45 | @Override 46 | public boolean interceptor(String url) { 47 | return true; 48 | } 49 | }); 50 | 51 | WebViewCacheInterceptorInst.getInstance(). 52 | init(builder); 53 | 54 | QbSdk.PreInitCallback cb = new QbSdk.PreInitCallback() { 55 | 56 | @Override 57 | public void onViewInitFinished(boolean arg0) { 58 | // TODO Auto-generated method stub 59 | //x5內核初始化完成的回调,为true表示x5内核加载成功,否则表示x5内核加载失败,会自动切换到系统内核。 60 | Log.d("app", " onViewInitFinished is " + arg0); 61 | } 62 | 63 | @Override 64 | public void onCoreInitFinished() { 65 | // TODO Auto-generated method stub 66 | } 67 | }; 68 | //x5内核初始化接口 69 | QbSdk.initX5Environment(getApplicationContext(), cb); 70 | 71 | 72 | 73 | } 74 | } 75 | -------------------------------------------------------------------------------- /app/src/main/java/ren/yale/android/cachewebview/JSInterface.java: -------------------------------------------------------------------------------- 1 | package ren.yale.android.cachewebview; 2 | 3 | import android.content.Context; 4 | import android.webkit.JavascriptInterface; 5 | import android.widget.Toast; 6 | 7 | /** 8 | * Created by yale on 2018/7/11. 9 | */ 10 | public class JSInterface { 11 | 12 | private Context mContext; 13 | 14 | public JSInterface(Context context) { 15 | mContext = context; 16 | } 17 | 18 | @JavascriptInterface 19 | public void toast(String text) { 20 | Toast.makeText(mContext, text, Toast.LENGTH_SHORT).show(); 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /app/src/main/java/ren/yale/android/cachewebview/MainActivity.java: -------------------------------------------------------------------------------- 1 | package ren.yale.android.cachewebview; 2 | 3 | import android.annotation.TargetApi; 4 | import android.app.Activity; 5 | import android.os.Build; 6 | import android.os.Bundle; 7 | import android.support.annotation.Nullable; 8 | import android.support.annotation.RequiresApi; 9 | import android.view.View; 10 | import android.webkit.CookieManager; 11 | import android.webkit.WebResourceError; 12 | import android.webkit.WebResourceRequest; 13 | import android.webkit.WebResourceResponse; 14 | import android.webkit.WebSettings; 15 | import android.webkit.WebView; 16 | import android.webkit.WebViewClient; 17 | import android.widget.AdapterView; 18 | import android.widget.CheckBox; 19 | import android.widget.CompoundButton; 20 | import android.widget.Spinner; 21 | 22 | import java.io.InputStream; 23 | 24 | import ren.yale.android.cachewebviewlib.WebViewCacheInterceptorInst; 25 | 26 | public class MainActivity extends Activity { 27 | 28 | private WebView mWebView; 29 | private static final String TAG = "CacheWebView"; 30 | private String URL = ""; 31 | 32 | 33 | @Override 34 | protected void onCreate(Bundle savedInstanceState) { 35 | super.onCreate(savedInstanceState); 36 | setContentView(R.layout.activity_main3); 37 | 38 | mWebView = findViewById(R.id.webview); 39 | 40 | CheckBox checkBox = (CheckBox) findViewById(R.id.checkbox); 41 | checkBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { 42 | @Override 43 | public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 44 | 45 | WebViewCacheInterceptorInst.getInstance().enableForce(isChecked); 46 | } 47 | }); 48 | final String[] urls = getResources().getStringArray(R.array.urls); 49 | URL = urls[0]; 50 | //URL=URL+"?r="+System.currentTimeMillis(); 51 | Spinner spinner = (Spinner) findViewById(R.id.spnner); 52 | spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { 53 | @Override 54 | public void onItemSelected(AdapterView parent, View view, int position, long id) { 55 | URL = urls[position]; 56 | //mInterceptor.loadUrl(mWebView,URL); 57 | //URL=URL+"?r="+System.currentTimeMillis(); 58 | WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,URL); 59 | } 60 | 61 | @Override 62 | public void onNothingSelected(AdapterView parent) { 63 | 64 | } 65 | }); 66 | initSettings(); 67 | mWebView.setWebViewClient(new WebViewClient(){ 68 | 69 | 70 | @Override 71 | public void onPageFinished(WebView view, String url) { 72 | super.onPageFinished(view, url); 73 | } 74 | 75 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 76 | @Override 77 | public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) { 78 | WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,request.getUrl().toString()); 79 | return true; 80 | } 81 | 82 | @Override 83 | public boolean shouldOverrideUrlLoading(WebView view, String url) { 84 | WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,url); 85 | return true; 86 | } 87 | 88 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 89 | @Nullable 90 | @Override 91 | public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) { 92 | return WebViewCacheInterceptorInst.getInstance().interceptRequest( request); 93 | } 94 | 95 | @Nullable 96 | @Override 97 | public WebResourceResponse shouldInterceptRequest(WebView view, String url) { 98 | return WebViewCacheInterceptorInst.getInstance().interceptRequest(url); 99 | } 100 | 101 | @Override 102 | public void onLoadResource(WebView view, String url) { 103 | super.onLoadResource(view, url); 104 | } 105 | 106 | @RequiresApi(api = Build.VERSION_CODES.M) 107 | @Override 108 | public void onReceivedError(WebView view, WebResourceRequest request, WebResourceError error) { 109 | int code = error.getErrorCode(); 110 | String resp = error.getDescription().toString(); 111 | String url = request.getUrl().toString(); 112 | super.onReceivedError(view, request, error); 113 | } 114 | }); 115 | //WebViewCacheInterceptorInst.getInstance().initAssetsData(); 116 | //WebViewCacheInterceptorInst.getInstance().loadUrl(mWebView,URL); 117 | 118 | } 119 | 120 | private void initSettings() { 121 | WebSettings webSettings = mWebView.getSettings(); 122 | 123 | webSettings.setJavaScriptEnabled(true); 124 | 125 | webSettings.setDomStorageEnabled(true); 126 | webSettings.setAllowFileAccess(true); 127 | webSettings.setUseWideViewPort(true); 128 | webSettings.setLoadWithOverviewMode(true); 129 | webSettings.setSupportZoom(true); 130 | webSettings.setBuiltInZoomControls(false); 131 | webSettings.setDisplayZoomControls(false); 132 | 133 | webSettings.setDefaultTextEncodingName("UTF-8"); 134 | 135 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { 136 | webSettings.setAllowFileAccessFromFileURLs(true); 137 | webSettings.setAllowUniversalAccessFromFileURLs(true); 138 | } 139 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 140 | CookieManager cookieManager = CookieManager.getInstance(); 141 | } 142 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 143 | webSettings.setMixedContentMode( 144 | WebSettings.MIXED_CONTENT_COMPATIBILITY_MODE); 145 | } 146 | } 147 | 148 | public void cleanCache(View v){ 149 | WebViewCacheInterceptorInst.getInstance().clearCache(); 150 | } 151 | public void getCacheFile(View v){ 152 | 153 | String url = "http://m.mm131.com/css/at.js"; 154 | InputStream inputStream = WebViewCacheInterceptorInst.getInstance().getCacheFile(url); 155 | if (inputStream!=null){ 156 | 157 | } 158 | 159 | } 160 | 161 | @Override 162 | public void onBackPressed() { 163 | if(mWebView.canGoBack()){ 164 | mWebView.goBack(); 165 | return; 166 | } 167 | // mInterceptor.clearCache(); 168 | super.onBackPressed(); 169 | } 170 | } 171 | -------------------------------------------------------------------------------- /app/src/main/java/ren/yale/android/cachewebview/WebResourceRequestAdapter.java: -------------------------------------------------------------------------------- 1 | package ren.yale.android.cachewebview; 2 | 3 | import android.annotation.TargetApi; 4 | import android.net.Uri; 5 | import android.os.Build; 6 | 7 | import java.util.Map; 8 | 9 | /** 10 | * Created by yale on 2018/7/26. 11 | */ 12 | @TargetApi(Build.VERSION_CODES.LOLLIPOP) 13 | public class WebResourceRequestAdapter implements android.webkit.WebResourceRequest { 14 | 15 | private com.tencent.smtt.export.external.interfaces.WebResourceRequest mWebResourceRequest; 16 | 17 | private WebResourceRequestAdapter(com.tencent.smtt.export.external.interfaces.WebResourceRequest x5Request){ 18 | mWebResourceRequest = x5Request; 19 | } 20 | 21 | public static WebResourceRequestAdapter adapter(com.tencent.smtt.export.external.interfaces.WebResourceRequest x5Request){ 22 | return new WebResourceRequestAdapter(x5Request); 23 | } 24 | 25 | @Override 26 | public Uri getUrl() { 27 | return mWebResourceRequest.getUrl(); 28 | } 29 | 30 | @Override 31 | public boolean isForMainFrame() { 32 | return mWebResourceRequest.isForMainFrame(); 33 | } 34 | 35 | @Override 36 | public boolean isRedirect() { 37 | return mWebResourceRequest.isRedirect(); 38 | } 39 | 40 | @Override 41 | public boolean hasGesture() { 42 | return mWebResourceRequest.hasGesture(); 43 | } 44 | 45 | @Override 46 | public String getMethod() { 47 | return mWebResourceRequest.getMethod(); 48 | } 49 | 50 | @Override 51 | public Map getRequestHeaders() { 52 | return mWebResourceRequest.getRequestHeaders(); 53 | } 54 | } 55 | -------------------------------------------------------------------------------- /app/src/main/java/ren/yale/android/cachewebview/WebResourceResponseAdapter.java: -------------------------------------------------------------------------------- 1 | package ren.yale.android.cachewebview; 2 | 3 | import android.os.Build; 4 | import android.support.annotation.RequiresApi; 5 | 6 | import java.io.InputStream; 7 | import java.util.Map; 8 | 9 | /** 10 | * Created by yale on 2018/7/26. 11 | */ 12 | public class WebResourceResponseAdapter extends com.tencent.smtt.export.external.interfaces.WebResourceResponse { 13 | 14 | private android.webkit.WebResourceResponse mWebResourceResponse; 15 | 16 | private WebResourceResponseAdapter(android.webkit.WebResourceResponse webResourceResponse){ 17 | mWebResourceResponse = webResourceResponse; 18 | } 19 | 20 | public static WebResourceResponseAdapter adapter(android.webkit.WebResourceResponse webResourceResponse){ 21 | if (webResourceResponse == null){ 22 | return null; 23 | } 24 | return new WebResourceResponseAdapter(webResourceResponse); 25 | 26 | } 27 | 28 | @Override 29 | public String getMimeType() { 30 | return mWebResourceResponse.getMimeType(); 31 | } 32 | 33 | @Override 34 | public InputStream getData() { 35 | return mWebResourceResponse.getData(); 36 | } 37 | 38 | @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) 39 | @Override 40 | public int getStatusCode() { 41 | return mWebResourceResponse.getStatusCode(); 42 | } 43 | 44 | @RequiresApi(Build.VERSION_CODES.LOLLIPOP) 45 | @Override 46 | public Map getResponseHeaders() { 47 | return mWebResourceResponse.getResponseHeaders(); 48 | } 49 | 50 | @Override 51 | public String getEncoding() { 52 | return mWebResourceResponse.getEncoding(); 53 | } 54 | 55 | @RequiresApi(Build.VERSION_CODES.LOLLIPOP) 56 | @Override 57 | public String getReasonPhrase() { 58 | return mWebResourceResponse.getReasonPhrase(); 59 | } 60 | } 61 | -------------------------------------------------------------------------------- /app/src/main/java/ren/yale/android/cachewebview/X5Activity.java: -------------------------------------------------------------------------------- 1 | package ren.yale.android.cachewebview; 2 | 3 | import android.app.Activity; 4 | import android.os.Bundle; 5 | import android.view.KeyEvent; 6 | import android.view.View; 7 | import android.view.ViewGroup; 8 | import android.widget.FrameLayout; 9 | 10 | import com.tencent.smtt.export.external.interfaces.IX5WebChromeClient; 11 | import com.tencent.smtt.export.external.interfaces.JsResult; 12 | import com.tencent.smtt.export.external.interfaces.WebResourceRequest; 13 | import com.tencent.smtt.export.external.interfaces.WebResourceResponse; 14 | import com.tencent.smtt.sdk.CookieSyncManager; 15 | import com.tencent.smtt.sdk.WebChromeClient; 16 | import com.tencent.smtt.sdk.WebSettings; 17 | import com.tencent.smtt.sdk.WebView; 18 | import com.tencent.smtt.sdk.WebViewClient; 19 | import com.tencent.smtt.utils.TbsLog; 20 | 21 | import ren.yale.android.cachewebview.x5.X5WebView; 22 | import ren.yale.android.cachewebviewlib.WebViewCacheInterceptorInst; 23 | 24 | public class X5Activity extends Activity { 25 | 26 | private ViewGroup mViewParent; 27 | private X5WebView mWebView; 28 | @Override 29 | protected void onCreate(Bundle savedInstanceState) { 30 | super.onCreate(savedInstanceState); 31 | setContentView(R.layout.activity_x5); 32 | 33 | mViewParent = (ViewGroup) findViewById(R.id.webView1); 34 | init(); 35 | } 36 | 37 | private void init() { 38 | 39 | mWebView = new X5WebView(this, null); 40 | 41 | mViewParent.addView(mWebView, new FrameLayout.LayoutParams( 42 | FrameLayout.LayoutParams.FILL_PARENT, 43 | FrameLayout.LayoutParams.FILL_PARENT)); 44 | 45 | 46 | mWebView.setWebViewClient(new WebViewClient() { 47 | @Override 48 | public boolean shouldOverrideUrlLoading(WebView view, String url) { 49 | if (url.startsWith("http")){ 50 | return false; 51 | } 52 | return true; 53 | } 54 | 55 | @Override 56 | public WebResourceResponse shouldInterceptRequest(WebView webView, String s) { 57 | return WebResourceResponseAdapter.adapter(WebViewCacheInterceptorInst.getInstance(). 58 | interceptRequest(s)); 59 | } 60 | 61 | @Override 62 | public WebResourceResponse shouldInterceptRequest(WebView webView, WebResourceRequest webResourceRequest) { 63 | 64 | return WebResourceResponseAdapter.adapter(WebViewCacheInterceptorInst.getInstance(). 65 | interceptRequest(WebResourceRequestAdapter.adapter(webResourceRequest))); 66 | } 67 | 68 | }); 69 | 70 | mWebView.setWebChromeClient(new WebChromeClient() { 71 | 72 | @Override 73 | public boolean onJsConfirm(WebView arg0, String arg1, String arg2, 74 | JsResult arg3) { 75 | return super.onJsConfirm(arg0, arg1, arg2, arg3); 76 | } 77 | 78 | View myVideoView; 79 | View myNormalView; 80 | IX5WebChromeClient.CustomViewCallback callback; 81 | 82 | // ///////////////////////////////////////////////////////// 83 | // 84 | /** 85 | * 全屏播放配置 86 | */ 87 | @Override 88 | public void onShowCustomView(View view, 89 | IX5WebChromeClient.CustomViewCallback customViewCallback) { 90 | 91 | } 92 | 93 | @Override 94 | public void onHideCustomView() { 95 | if (callback != null) { 96 | callback.onCustomViewHidden(); 97 | callback = null; 98 | } 99 | if (myVideoView != null) { 100 | ViewGroup viewGroup = (ViewGroup) myVideoView.getParent(); 101 | viewGroup.removeView(myVideoView); 102 | viewGroup.addView(myNormalView); 103 | } 104 | } 105 | 106 | @Override 107 | public boolean onJsAlert(WebView arg0, String arg1, String arg2, 108 | JsResult arg3) { 109 | /** 110 | * 这里写入你自定义的window alert 111 | */ 112 | return super.onJsAlert(null, arg1, arg2, arg3); 113 | } 114 | }); 115 | 116 | WebSettings webSetting = mWebView.getSettings(); 117 | webSetting.setAllowFileAccess(true); 118 | webSetting.setLayoutAlgorithm(WebSettings.LayoutAlgorithm.NARROW_COLUMNS); 119 | webSetting.setSupportZoom(true); 120 | webSetting.setBuiltInZoomControls(true); 121 | webSetting.setUseWideViewPort(true); 122 | webSetting.setSupportMultipleWindows(false); 123 | // webSetting.setLoadWithOverviewMode(true); 124 | webSetting.setAppCacheEnabled(true); 125 | // webSetting.setDatabaseEnabled(true); 126 | webSetting.setDomStorageEnabled(true); 127 | webSetting.setJavaScriptEnabled(true); 128 | webSetting.setGeolocationEnabled(true); 129 | webSetting.setAppCacheMaxSize(Long.MAX_VALUE); 130 | webSetting.setAppCachePath(this.getDir("appcache", 0).getPath()); 131 | webSetting.setDatabasePath(this.getDir("databases", 0).getPath()); 132 | webSetting.setGeolocationDatabasePath(this.getDir("geolocation", 0) 133 | .getPath()); 134 | // webSetting.setPageCacheCapacity(IX5WebSettings.DEFAULT_CACHE_CAPACITY); 135 | webSetting.setPluginState(WebSettings.PluginState.ON_DEMAND); 136 | // webSetting.setRenderPriority(WebSettings.RenderPriority.HIGH); 137 | // webSetting.setPreFectch(true); 138 | long time = System.currentTimeMillis(); 139 | String url = "https://www.baidu.com"; 140 | 141 | mWebView.loadUrl(url); 142 | WebViewCacheInterceptorInst.getInstance().loadUrl(url,mWebView.getSettings().getUserAgentString()); 143 | 144 | TbsLog.d("time-cost", "cost time: " 145 | + (System.currentTimeMillis() - time)); 146 | CookieSyncManager.createInstance(this); 147 | CookieSyncManager.getInstance().sync(); 148 | } 149 | 150 | @Override 151 | public boolean onKeyDown(int keyCode, KeyEvent event) { 152 | 153 | if (keyCode == KeyEvent.KEYCODE_BACK) { 154 | if (mWebView != null && mWebView.canGoBack()) { 155 | mWebView.goBack(); 156 | return true; 157 | } else 158 | return super.onKeyDown(keyCode, event); 159 | } 160 | return super.onKeyDown(keyCode, event); 161 | } 162 | } 163 | -------------------------------------------------------------------------------- /app/src/main/java/ren/yale/android/cachewebview/x5/X5WebView.java: -------------------------------------------------------------------------------- 1 | package ren.yale.android.cachewebview.x5; 2 | 3 | import android.annotation.SuppressLint; 4 | import android.content.Context; 5 | import android.graphics.Canvas; 6 | import android.graphics.Paint; 7 | import android.os.Build; 8 | import android.util.AttributeSet; 9 | import android.view.View; 10 | import android.widget.TextView; 11 | 12 | import com.tencent.smtt.export.external.interfaces.WebResourceResponse; 13 | import com.tencent.smtt.sdk.QbSdk; 14 | import com.tencent.smtt.sdk.WebSettings; 15 | import com.tencent.smtt.sdk.WebSettings.LayoutAlgorithm; 16 | import com.tencent.smtt.sdk.WebView; 17 | import com.tencent.smtt.sdk.WebViewClient; 18 | 19 | public class X5WebView extends WebView { 20 | TextView title; 21 | private WebViewClient client = new WebViewClient() { 22 | /** 23 | * 防止加载网页时调起系统浏览器 24 | */ 25 | public boolean shouldOverrideUrlLoading(WebView view, String url) { 26 | view.loadUrl(url); 27 | return true; 28 | } 29 | 30 | @Override 31 | public WebResourceResponse shouldInterceptRequest(WebView webView, String s) { 32 | return super.shouldInterceptRequest(webView, s); 33 | } 34 | }; 35 | 36 | @SuppressLint("SetJavaScriptEnabled") 37 | public X5WebView(Context arg0, AttributeSet arg1) { 38 | super(arg0, arg1); 39 | //this.setWebViewClient(client); 40 | // this.setWebChromeClient(chromeClient); 41 | // WebStorage webStorage = WebStorage.getInstance(); 42 | initWebViewSettings(); 43 | this.getView().setClickable(true); 44 | } 45 | 46 | private void initWebViewSettings() { 47 | WebSettings webSetting = this.getSettings(); 48 | webSetting.setJavaScriptEnabled(true); 49 | webSetting.setJavaScriptCanOpenWindowsAutomatically(true); 50 | webSetting.setAllowFileAccess(true); 51 | webSetting.setLayoutAlgorithm(LayoutAlgorithm.NARROW_COLUMNS); 52 | webSetting.setSupportZoom(true); 53 | webSetting.setBuiltInZoomControls(true); 54 | webSetting.setUseWideViewPort(true); 55 | webSetting.setSupportMultipleWindows(true); 56 | // webSetting.setLoadWithOverviewMode(true); 57 | webSetting.setAppCacheEnabled(true); 58 | // webSetting.setDatabaseEnabled(true); 59 | webSetting.setDomStorageEnabled(true); 60 | webSetting.setGeolocationEnabled(true); 61 | webSetting.setAppCacheMaxSize(Long.MAX_VALUE); 62 | // webSetting.setPageCacheCapacity(IX5WebSettings.DEFAULT_CACHE_CAPACITY); 63 | webSetting.setPluginState(WebSettings.PluginState.ON_DEMAND); 64 | // webSetting.setRenderPriority(WebSettings.RenderPriority.HIGH); 65 | webSetting.setCacheMode(WebSettings.LOAD_NO_CACHE); 66 | 67 | // this.getSettingsExtension().setPageCacheCapacity(IX5WebSettings.DEFAULT_CACHE_CAPACITY);//extension 68 | // settings 的设计 69 | } 70 | 71 | @Override 72 | protected boolean drawChild(Canvas canvas, View child, long drawingTime) { 73 | boolean ret = super.drawChild(canvas, child, drawingTime); 74 | canvas.save(); 75 | Paint paint = new Paint(); 76 | paint.setColor(0x7fff0000); 77 | paint.setTextSize(24.f); 78 | paint.setAntiAlias(true); 79 | if (getX5WebViewExtension() != null) { 80 | canvas.drawText(this.getContext().getPackageName() + "-pid:" 81 | + android.os.Process.myPid(), 10, 50, paint); 82 | canvas.drawText( 83 | "X5 Core:" + QbSdk.getTbsVersion(this.getContext()), 10, 84 | 100, paint); 85 | } else { 86 | canvas.drawText(this.getContext().getPackageName() + "-pid:" 87 | + android.os.Process.myPid(), 10, 50, paint); 88 | canvas.drawText("Sys Core", 10, 100, paint); 89 | } 90 | canvas.drawText(Build.MANUFACTURER, 10, 150, paint); 91 | canvas.drawText(Build.MODEL, 10, 200, paint); 92 | canvas.restore(); 93 | return ret; 94 | } 95 | 96 | public X5WebView(Context arg0) { 97 | super(arg0); 98 | setBackgroundColor(85621); 99 | } 100 | 101 | } 102 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 6 | 7 | 11 | 12 | 18 | 19 |