app内打开浏览器目前主要2个cordova插件
cordova-plugin-inappbrowser 和 cordova-plugin-themeablebrowser
themeablebrowser是在cordova-plugin-inappbrowser基础上二次开发的, 支持webview中有按钮,及相关按钮事件 是我们想要的
第一步
在已有项目目录下添加插件
cordova plugin add cordova-plugin-themeablebrowser
第二步 准备图片
vue项目下新建static/browserIcons文件夹, 放入准备好的 back.png 和 close.png
第三步 修改webpack打包规则
根据插件要求, wwwImage
从Cordova的www
目录加载图像。所以我们需要把插件需要的第二步图片 放到cordova www目录下
因为www目录 我们之前修改过vue项目的build目录 每次vue build 会把www目录删掉从新生成, 所以手工拷贝不靠谱
安装 copy-webpack-plugin
npm install --save copy-webpack-plugin
修改 vue项目 vue.config.js
from 定义要拷贝的源文件 from:__dirname+'/src/components'
to 定义要拷贝到的目标文件夹 to: __dirname+'/dist'
toType file 或者 dir 可选,默认是文件
force 强制覆盖前面的插件 可选,默认是文件
context 可选,默认base context可用specific context
flatten 只拷贝指定的文件 可以用模糊匹配
ignore 忽略拷贝指定的文件 可以模糊匹配
第四步 修改相应vue页面 我这里是修改的 helloword.vue
相关按钮及事件 参考 插件github https://github.com/initialxy/cordova-plugin-themeablebrowser
<template>
<div class="hello">
<h1>{{ msg }}</h1>
<button @click="handleAxiosRequest">axios调用</button>
<h1>{{ bjtime }}</h1>
<button @click="openAppBrowser('https://www.google.com/')">跳转页面1</button>
<button @click="openAppBrowser('https://www.nginx.com/')">跳转页面2</button>
</div>
</template>
<script>
export default {
name: "HelloWorld",
props: {
msg: String,
},
data() {
return {
bjtime: null,
};
},
methods: {
openAppBrowser(url) {
// Keep in mind that you must add your own images to native resource.
// Images below are for sample only. They are not imported by this plugin.
window.cordova.ThemeableBrowser.open(url, "_blank", {
statusbar: {
color: "#EBCE9C",
},
title: {
color: "#000000",
showPageTitle: true,
staticText:"换成自己的标题,或者不显示"
},
backButton: {
wwwImage: "browserIcons/back.png",
wwwImagePressed: "browserIcons/back.png",
align: "left",
wwwImageDensity: 2,
event: "backPressed",
},
closeButton: {
wwwImage: "browserIcons/close.png",
wwwImagePressed: "browserIcons/close.png",
align: "right",
wwwImageDensity: 2,
event: "closePressed",
},
backButtonCanClose: true,
})
.addEventListener("backPressed", function (e) {
alert("back pressed"+e);
})
.addEventListener("helloPressed", function (e) {
alert("hello pressed"+e);
})
.addEventListener("sharePressed", function (e) {
alert(e.url);
})
// .addEventListener(window.cordova.ThemeableBrowser.EVT_ERR, function (e) {
// console.error(e.message);
// })
// .addEventListener(window.cordova.ThemeableBrowser.EVT_WRN, function (e) {
// console.log(e.message);
// });
},
handleAxiosRequest() {
this.$axios
.get("https://quan.suning.com/getSysTime.do")
.then(({ data }) => {
this.bjtime = data;
})
.catch((err) => {
console.log(err);
});
},
},
};
</script>
就可以看到页面了 点击会返回和关闭 会触发设定好的事件
第五步 修改源码支持横竖屏打开
cordova-plugin-themeablebrowser这个插件 并不支持横竖屏打开webview页面 如何让它支持呢
用 Android studio打开 platforms/android/build.gradle 文件 找到下面 ThemeableBrowser 类
把代码贴出来
1 /*
2 Licensed to the Apache Software Foundation (ASF) under one
3 or more contributor license agreements. See the NOTICE file
4 distributed with this work for additional information
5 regarding copyright ownership. The ASF licenses this file
6 to you under the Apache License, Version 2.0 (the
7 "License"); you may not use this file except in compliance
8 with the License. You may obtain a copy of the License at
9
10 http://www.apache.org/licenses/LICENSE-2.0
11
12 Unless required by applicable law or agreed to in writing,
13 software distributed under the License is distributed on an
14 "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 KIND, either express or implied. See the License for the
16 specific language governing permissions and limitations
17 under the License.
18 */
19 package com.initialxy.cordova.themeablebrowser;
20
21 import android.annotation.SuppressLint;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.content.res.Resources;
25 import android.graphics.Bitmap;
26 import android.graphics.BitmapFactory;
27 import android.graphics.Canvas;
28 import android.graphics.Paint;
29 import android.graphics.drawable.BitmapDrawable;
30 import android.graphics.drawable.Drawable;
31 import android.graphics.drawable.StateListDrawable;
32 import android.net.Uri;
33 import android.os.Build;
34 import android.os.Bundle;
35 import android.provider.Browser;
36 import android.text.InputType;
37 import android.text.TextUtils;
38 import android.util.DisplayMetrics;
39 import android.util.Log;
40 import android.util.TypedValue;
41 import android.view.Gravity;
42 import android.view.KeyEvent;
43 import android.view.MotionEvent;
44 import android.view.View;
45 import android.view.ViewGroup;
46 import android.view.ViewGroup.LayoutParams;
47 import android.view.Window;
48 import android.view.WindowManager;
49 import android.view.inputmethod.EditorInfo;
50 import android.view.inputmethod.InputMethodManager;
51 import android.webkit.CookieManager;
52 import android.webkit.WebSettings;
53 import android.webkit.WebView;
54 import android.webkit.WebViewClient;
55 import android.widget.AdapterView;
56 import android.widget.ArrayAdapter;
57 import android.widget.Button;
58 import android.widget.EditText;
59 import android.widget.FrameLayout;
60 import android.widget.LinearLayout;
61 import android.widget.RelativeLayout;
62 import android.widget.Spinner;
63 import android.widget.TextView;
64
65 import org.apache.cordova.CallbackContext;
66 import org.apache.cordova.CordovaArgs;
67 import org.apache.cordova.CordovaPlugin;
68 import org.apache.cordova.CordovaWebView;
69 import org.apache.cordova.PluginManager;
70 import org.apache.cordova.PluginResult;
71 import org.apache.cordova.Whitelist;
72 import org.json.JSONException;
73 import org.json.JSONObject;
74
75 import java.io.File;
76 import java.io.IOException;
77 import java.io.InputStream;
78 import java.lang.reflect.InvocationTargetException;
79 import java.lang.reflect.Method;
80
81 @SuppressLint("SetJavaScriptEnabled")
82 public class ThemeableBrowser extends CordovaPlugin {
83
84 private static final String NULL = "null";
85 protected static final String LOG_TAG = "ThemeableBrowser";
86 private static final String SELF = "_self";
87 private static final String SYSTEM = "_system";
88 // private static final String BLANK = "_blank";
89 private static final String EXIT_EVENT = "exit";
90 private static final String LOAD_START_EVENT = "loadstart";
91 private static final String LOAD_STOP_EVENT = "loadstop";
92 private static final String LOAD_ERROR_EVENT = "loaderror";
93
94 private static final String ALIGN_LEFT = "left";
95 private static final String ALIGN_RIGHT = "right";
96
97 private static final int TOOLBAR_DEF_HEIGHT = 44;
98 private static final int DISABLED_ALPHA = 127; // 50% AKA 127/255.
99
100 private static final String EVT_ERR = "ThemeableBrowserError";
101 private static final String EVT_WRN = "ThemeableBrowserWarning";
102 private static final String ERR_CRITICAL = "critical";
103 private static final String ERR_LOADFAIL = "loadfail";
104 private static final String WRN_UNEXPECTED = "unexpected";
105 private static final String WRN_UNDEFINED = "undefined";
106
107 private ThemeableBrowserDialog dialog;
108 private WebView inAppWebView;
109 private EditText edittext;
110 private CallbackContext callbackContext;
111
112 /**
113 * Executes the request and returns PluginResult.
114 *
115 * @param action The action to execute.
116 * @param args The exec() arguments, wrapped with some Cordova helpers.
117 * @param callbackContext The callback context used when calling back into JavaScript.
118 * @return
119 * @throws JSONException
120 */
121 public boolean execute(String action, CordovaArgs args, final CallbackContext callbackContext) throws JSONException {
122 if (action.equals("open")) {
123 this.callbackContext = callbackContext;
124 final String url = args.getString(0);
125 String t = args.optString(1);
126 if (t == null || t.equals("") || t.equals(NULL)) {
127 t = SELF;
128 }
129 final String target = t;
130 final Options features = parseFeature(args.optString(2));
131
132 this.cordova.getActivity().runOnUiThread(new Runnable() {
133 @Override
134 public void run() {
135 String result = "";
136 // SELF
137 if (SELF.equals(target)) {
138 /* This code exists for compatibility between 3.x and 4.x versions of Cordova.
139 * Previously the Config class had a static method, isUrlWhitelisted(). That
140 * responsibility has been moved to the plugins, with an aggregating method in
141 * PluginManager.
142 */
143 Boolean shouldAllowNavigation = null;
144 if (url.startsWith("javascript:")) {
145 shouldAllowNavigation = true;
146 }
147 if (shouldAllowNavigation == null) {
148 shouldAllowNavigation = new Whitelist().isUrlWhiteListed(url);
149 }
150 if (shouldAllowNavigation == null) {
151 try {
152 Method gpm = webView.getClass().getMethod("getPluginManager");
153 PluginManager pm = (PluginManager)gpm.invoke(webView);
154 Method san = pm.getClass().getMethod("shouldAllowNavigation", String.class);
155 shouldAllowNavigation = (Boolean)san.invoke(pm, url);
156 } catch (NoSuchMethodException e) {
157 } catch (IllegalAccessException e) {
158 } catch (InvocationTargetException e) {
159 }
160 }
161 // load in webview
162 if (Boolean.TRUE.equals(shouldAllowNavigation)) {
163 webView.loadUrl(url);
164 }
165 //Load the dialer
166 else if (url.startsWith(WebView.SCHEME_TEL))
167 {
168 try {
169 Intent intent = new Intent(Intent.ACTION_DIAL);
170 intent.setData(Uri.parse(url));
171 cordova.getActivity().startActivity(intent);
172 } catch (android.content.ActivityNotFoundException e) {
173 emitError(ERR_CRITICAL,
174 String.format("Error dialing %s: %s", url, e.toString()));
175 }
176 }
177 // load in ThemeableBrowser
178 else {
179 result = showWebPage(url, features);
180 }
181 }
182 // SYSTEM
183 else if (SYSTEM.equals(target)) {
184 result = openExternal(url);
185 }
186 // BLANK - or anything else
187 else {
188 result = showWebPage(url, features);
189 }
190
191 PluginResult pluginResult = new PluginResult(PluginResult.Status.OK, result);
192 pluginResult.setKeepCallback(true);
193 callbackContext.sendPluginResult(pluginResult);
194 }
195 });
196 }
197 else if (action.equals("close")) {
198 closeDialog();
199 }
200 else if (action.equals("injectScriptCode")) {
201 String jsWrapper = null;
202 if (args.getBoolean(1)) {
203 jsWrapper = String.format("prompt(JSON.stringify([eval(%%s)]), 'gap-iab://%s')", callbackContext.getCallbackId());
204 }
205 injectDeferredObject(args.getString(0), jsWrapper);
206 }
207 else if (action.equals("injectScriptFile")) {
208 String jsWrapper;
209 if (args.getBoolean(1)) {
210 jsWrapper = String.format("(function(d) { var c = d.createElement('script'); c.src = %%s; c.onload = function() { prompt('', 'gap-iab://%s'); }; d.body.appendChild(c); })(document)", callbackContext.getCallbackId());
211 } else {
212 jsWrapper = "(function(d) { var c = d.createElement('script'); c.src = %s; d.body.appendChild(c); })(document)";
213 }
214 injectDeferredObject(args.getString(0), jsWrapper);
215 }
216 else if (action.equals("injectStyleCode")) {
217 String jsWrapper;
218 if (args.getBoolean(1)) {
219 jsWrapper = String.format("(function(d) { var c = d.createElement('style'); c.innerHTML = %%s; d.body.appendChild(c); prompt('', 'gap-iab://%s');})(document)", callbackContext.getCallbackId());
220 } else {
221 jsWrapper = "(function(d) { var c = d.createElement('style'); c.innerHTML = %s; d.body.appendChild(c); })(document)";
222 }
223 injectDeferredObject(args.getString(0), jsWrapper);
224 }
225 else if (action.equals("injectStyleFile")) {
226 String jsWrapper;
227 if (args.getBoolean(1)) {
228 jsWrapper = String.format("(function(d) { var c = d.createElement('link'); c.rel='stylesheet'; c.type='text/css'; c.href = %%s; d.head.appendChild(c); prompt('', 'gap-iab://%s');})(document)", callbackContext.getCallbackId());
229 } else {
230 jsWrapper = "(function(d) { var c = d.createElement('link'); c.rel='stylesheet'; c.type='text/css'; c.href = %s; d.head.appendChild(c); })(document)";
231 }
232 injectDeferredObject(args.getString(0), jsWrapper);
233 }
234 else if (action.equals("show")) {
235 this.cordova.getActivity().runOnUiThread(new Runnable() {
236 @Override
237 public void run() {
238 dialog.show();
239 }
240 });
241 PluginResult pluginResult = new PluginResult(PluginResult.Status.OK);
242 pluginResult.setKeepCallback(true);
243 this.callbackContext.sendPluginResult(pluginResult);
244 }
245 else if (action.equals("reload")) {
246 if (inAppWebView != null) {
247 this.cordova.getActivity().runOnUiThread(new Runnable() {
248 @Override
249 public void run() {
250 inAppWebView.reload();
251 }
252 });
253 }
254 }
255 else {
256 return false;
257 }
258 return true;
259 }
260
261 /**
262 * Called when the view navigates.
263 */
264 @Override
265 public void onReset() {
266 closeDialog();
267 }
268
269 /**
270 * Called by AccelBroker when listener is to be shut down.
271 * Stop listener.
272 */
273 public void onDestroy() {
274 closeDialog();
275 }
276
277 /**
278 * Inject an object (script or style) into the ThemeableBrowser WebView.
279 *
280 * This is a helper method for the inject{Script|Style}{Code|File} API calls, which
281 * provides a consistent method for injecting JavaScript code into the document.
282 *
283 * If a wrapper string is supplied, then the source string will be JSON-encoded (adding
284 * quotes) and wrapped using string formatting. (The wrapper string should have a single
285 * '%s' marker)
286 *
287 * @param source The source object (filename or script/style text) to inject into
288 * the document.
289 * @param jsWrapper A JavaScript string to wrap the source string in, so that the object
290 * is properly injected, or null if the source string is JavaScript text
291 * which should be executed directly.
292 */
293 private void injectDeferredObject(String source, String jsWrapper) {
294 String scriptToInject;
295 if (jsWrapper != null) {
296 org.json.JSONArray jsonEsc = new org.json.JSONArray();
297 jsonEsc.put(source);
298 String jsonRepr = jsonEsc.toString();
299 String jsonSourceString = jsonRepr.substring(1, jsonRepr.length()-1);
300 scriptToInject = String.format(jsWrapper, jsonSourceString);
301 } else {
302 scriptToInject = source;
303 }
304 final String finalScriptToInject = scriptToInject;
305 this.cordova.getActivity().runOnUiThread(new Runnable() {
306 @SuppressLint("NewApi")
307 @Override
308 public void run() {
309 if (inAppWebView != null) {
310 if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT) {
311 // This action will have the side-effect of blurring the currently focused
312 // element
313 inAppWebView.loadUrl("javascript:" + finalScriptToInject);
314 } else {
315 inAppWebView.evaluateJavascript(finalScriptToInject, null);
316 }
317 }
318 }
319 });
320 }
321
322 /**
323 * Put the list of features into a hash map
324 *
325 * @param optString
326 * @return
327 */
328 private Options parseFeature(String optString) {
329 Options result = null;
330 if (optString != null && !optString.isEmpty()) {
331 try {
332 result = ThemeableBrowserUnmarshaller.JSONToObj(
333 optString, Options.class);
334 } catch (Exception e) {
335 emitError(ERR_CRITICAL,
336 String.format("Invalid JSON @s", e.toString()));
337 }
338 } else {
339 emitWarning(WRN_UNDEFINED,
340 "No config was given, defaults will be used, "
341 + "which is quite boring.");
342 }
343
344 if (result == null) {
345 result = new Options();
346 }
347
348 // Always show location, this property is overwritten.
349 result.location = true;
350
351 return result;
352 }
353
354 /**
355 * Display a new browser with the specified URL.
356 *
357 * @param url
358 * @return
359 */
360 public String openExternal(String url) {
361 try {
362 Intent intent = null;
363 intent = new Intent(Intent.ACTION_VIEW);
364 // Omitting the MIME type for file: URLs causes "No Activity found to handle Intent".
365 // Adding the MIME type to http: URLs causes them to not be handled by the downloader.
366 Uri uri = Uri.parse(url);
367 if ("file".equals(uri.getScheme())) {
368 intent.setDataAndType(uri, webView.getResourceApi().getMimeType(uri));
369 } else {
370 intent.setData(uri);
371 }
372 intent.putExtra(Browser.EXTRA_APPLICATION_ID, cordova.getActivity().getPackageName());
373 this.cordova.getActivity().startActivity(intent);
374 return "";
375 } catch (android.content.ActivityNotFoundException e) {
376 Log.d(LOG_TAG, "ThemeableBrowser: Error loading url "+url+":"+ e.toString());
377 return e.toString();
378 }
379 }
380
381 /**
382 * Closes the dialog
383 */
384 public void closeDialog() {
385 this.cordova.getActivity().runOnUiThread(new Runnable() {
386 @Override
387 public void run() {
388 // The JS protects against multiple calls, so this should happen only when
389 // closeDialog() is called by other native code.
390 if (inAppWebView == null) {
391 emitWarning(WRN_UNEXPECTED, "Close called but already closed.");
392 return;
393 }
394
395 inAppWebView.setWebViewClient(new WebViewClient() {
396 // NB: wait for about:blank before dismissing
397 public void onPageFinished(WebView view, String url) {
398 if (dialog != null) {
399 dialog.dismiss();
400 }
401
402 // Clean up.
403 dialog = null;
404 inAppWebView = null;
405 edittext = null;
406 callbackContext = null;
407 }
408 });
409
410 // NB: From SDK 19: "If you call methods on WebView from any
411 // thread other than your app's UI thread, it can cause
412 // unexpected results."
413 // http://developer.android.com/guide/webapps/migrating.html#Threads
414 inAppWebView.loadUrl("about:blank");
415
416 try {
417 JSONObject obj = new JSONObject();
418 obj.put("type", EXIT_EVENT);
419 sendUpdate(obj, false);
420 } catch (JSONException ex) {
421 }
422 }
423 });
424 }
425
426 private void emitButtonEvent(Event event, String url) {
427 emitButtonEvent(event, url, null);
428 }
429
430 private void emitButtonEvent(Event event, String url, Integer index) {
431 if (event != null && event.event != null) {
432 try {
433 JSONObject obj = new JSONObject();
434 obj.put("type", event.event);
435 obj.put("url", url);
436 if (index != null) {
437 obj.put("index", index.intValue());
438 }
439 sendUpdate(obj, true);
440 } catch (JSONException e) {
441 // Ignore, should never happen.
442 }
443 } else {
444 emitWarning(WRN_UNDEFINED,
445 "Button clicked, but event property undefined. "
446 + "No event will be raised.");
447 }
448 }
449
450 private void emitError(String code, String message) {
451 emitLog(EVT_ERR, code, message);
452 }
453
454 private void emitWarning(String code, String message) {
455 emitLog(EVT_WRN, code, message);
456 }
457
458 private void emitLog(String type, String code, String message) {
459 if (type != null) {
460 try {
461 JSONObject obj = new JSONObject();
462 obj.put("type", type);
463 obj.put("code", code);
464 obj.put("message", message);
465 sendUpdate(obj, true);
466 } catch (JSONException e) {
467 // Ignore, should never happen.
468 }
469 }
470 }
471
472 /**
473 * Checks to see if it is possible to go back one page in history, then does so.
474 */
475 public void goBack() {
476 if (this.inAppWebView != null && this.inAppWebView.canGoBack()) {
477 this.inAppWebView.goBack();
478 }
479 }
480
481 /**
482 * Can the web browser go back?
483 * @return boolean
484 */
485 public boolean canGoBack() {
486 return this.inAppWebView != null && this.inAppWebView.canGoBack();
487 }
488
489 /**
490 * Checks to see if it is possible to go forward one page in history, then does so.
491 */
492 private void goForward() {
493 if (this.inAppWebView != null && this.inAppWebView.canGoForward()) {
494 this.inAppWebView.goForward();
495 }
496 }
497
498 /**
499 * Navigate to the new page
500 *
501 * @param url to load
502 */
503 private void navigate(String url) {
504 InputMethodManager imm = (InputMethodManager)this.cordova.getActivity().getSystemService(Context.INPUT_METHOD_SERVICE);
505 imm.hideSoftInputFromWindow(edittext.getWindowToken(), 0);
506
507 if (!url.startsWith("http") && !url.startsWith("file:")) {
508 this.inAppWebView.loadUrl("http://" + url);
509 } else {
510 this.inAppWebView.loadUrl(url);
511 }
512 this.inAppWebView.requestFocus();
513 }
514
515 private ThemeableBrowser getThemeableBrowser() {
516 return this;
517 }
518
519 /**
520 * Display a new browser with the specified URL.
521 *
522 * @param url
523 * @param features
524 * @return
525 */
526 public String showWebPage(final String url, final Options features) {
527 final CordovaWebView thatWebView = this.webView;
528
529 // Create dialog in new thread
530 Runnable runnable = new Runnable() {
531 @SuppressLint("NewApi")
532 public void run() {
533 // Let's create the main dialog
534 dialog = new ThemeableBrowserDialog(cordova.getActivity(),
535 android.R.style.Theme_Black_NoTitleBar,
536 features.hardwareback);
537 if (!features.disableAnimation) {
538 dialog.getWindow().getAttributes().windowAnimations
539 = android.R.style.Animation_Dialog;
540 }
541 dialog.requestWindowFeature(Window.FEATURE_NO_TITLE);
542 dialog.setCancelable(true);
543 dialog.setThemeableBrowser(getThemeableBrowser());
544
545 // Main container layout
546 ViewGroup main = null;
547
548 if (features.fullscreen) {
549 main = new FrameLayout(cordova.getActivity());
550 } else {
551 main = new LinearLayout(cordova.getActivity());
552 ((LinearLayout) main).setOrientation(LinearLayout.VERTICAL);
553 }
554
555 // Toolbar layout
556 Toolbar toolbarDef = features.toolbar;
557 FrameLayout toolbar = new FrameLayout(cordova.getActivity());
558 toolbar.setBackgroundColor(hexStringToColor(
559 toolbarDef != null && toolbarDef.color != null
560 ? toolbarDef.color : "#ffffffff"));
561 toolbar.setLayoutParams(new ViewGroup.LayoutParams(
562 LayoutParams.MATCH_PARENT,
563 dpToPixels(toolbarDef != null
564 ? toolbarDef.height : TOOLBAR_DEF_HEIGHT)));
565
566 if (toolbarDef != null
567 && (toolbarDef.image != null || toolbarDef.wwwImage != null)) {
568 try {
569 Drawable background = getImage(toolbarDef.image
570 , toolbarDef.wwwImage, toolbarDef.wwwImageDensity);
571 setBackground(toolbar, background);
572 } catch (Resources.NotFoundException e) {
573 emitError(ERR_LOADFAIL,
574 String.format("Image for toolbar, %s, failed to load",
575 toolbarDef.image));
576 } catch (IOException ioe) {
577 emitError(ERR_LOADFAIL,
578 String.format("Image for toolbar, %s, failed to load",
579 toolbarDef.wwwImage));
580 }
581 }
582
583 // Left Button Container layout
584 LinearLayout leftButtonContainer = new LinearLayout(cordova.getActivity());
585 FrameLayout.LayoutParams leftButtonContainerParams = new FrameLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
586 leftButtonContainerParams.gravity = Gravity.LEFT | Gravity.CENTER_VERTICAL;
587 leftButtonContainer.setLayoutParams(leftButtonContainerParams);
588 leftButtonContainer.setVerticalGravity(Gravity.CENTER_VERTICAL);
589
590 // Right Button Container layout
591 LinearLayout rightButtonContainer = new LinearLayout(cordova.getActivity());
592 FrameLayout.LayoutParams rightButtonContainerParams = new FrameLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
593 rightButtonContainerParams.gravity = Gravity.RIGHT | Gravity.CENTER_VERTICAL;
594 rightButtonContainer.setLayoutParams(rightButtonContainerParams);
595 rightButtonContainer.setVerticalGravity(Gravity.CENTER_VERTICAL);
596
597 // Edit Text Box
598 edittext = new EditText(cordova.getActivity());
599 RelativeLayout.LayoutParams textLayoutParams = new RelativeLayout.LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);
600 textLayoutParams.addRule(RelativeLayout.RIGHT_OF, 1);
601 textLayoutParams.addRule(RelativeLayout.LEFT_OF, 5);
602 edittext.setLayoutParams(textLayoutParams);
603 edittext.setSingleLine(true);
604 edittext.setText(url);
605 edittext.setInputType(InputType.TYPE_TEXT_VARIATION_URI);
606 edittext.setImeOptions(EditorInfo.IME_ACTION_GO);
607 edittext.setInputType(InputType.TYPE_NULL); // Will not except input... Makes the text NON-EDITABLE
608 edittext.setOnKeyListener(new View.OnKeyListener() {
609 public boolean onKey(View v, int keyCode, KeyEvent event) {
610 // If the event is a key-down event on the "enter" button
611 if ((event.getAction() == KeyEvent.ACTION_DOWN) && (keyCode == KeyEvent.KEYCODE_ENTER)) {
612 navigate(edittext.getText().toString());
613 return true;
614 }
615 return false;
616 }
617 });
618
619 // Back button
620 final Button back = createButton(
621 features.backButton,
622 "back button",
623 new View.OnClickListener() {
624 public void onClick(View v) {
625 emitButtonEvent(
626 features.backButton,
627 inAppWebView.getUrl());
628
629 if (features.backButtonCanClose && !canGoBack()) {
630 closeDialog();
631 } else {
632 goBack();
633 }
634 }
635 }
636 );
637
638 if (back != null) {
639 back.setEnabled(features.backButtonCanClose);
640 }
641
642 // Forward button
643 final Button forward = createButton(
644 features.forwardButton,
645 "forward button",
646 new View.OnClickListener() {
647 public void onClick(View v) {
648 emitButtonEvent(
649 features.forwardButton,
650 inAppWebView.getUrl());
651
652 goForward();
653 }
654 }
655 );
656
657 if (back != null) {
658 back.setEnabled(false);
659 }
660
661
662 // Close/Done button
663 Button close = createButton(
664 features.closeButton,
665 "close button",
666 new View.OnClickListener() {
667 public void onClick(View v) {
668 emitButtonEvent(
669 features.closeButton,
670 inAppWebView.getUrl());
671 closeDialog();
672 }
673 }
674 );
675
676 // Menu button
677 Spinner menu = features.menu != null
678 ? new MenuSpinner(cordova.getActivity()) : null;
679 if (menu != null) {
680 menu.setLayoutParams(new LinearLayout.LayoutParams(
681 LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));
682 menu.setContentDescription("menu button");
683 setButtonImages(menu, features.menu, DISABLED_ALPHA);
684
685 // We are not allowed to use onClickListener for Spinner, so we will use
686 // onTouchListener as a fallback.
687 menu.setOnTouchListener(new View.OnTouchListener() {
688 @Override
689 public boolean onTouch(View v, MotionEvent event) {
690 if (event.getAction() == MotionEvent.ACTION_UP) {
691 emitButtonEvent(
692 features.menu,
693 inAppWebView.getUrl());
694 }
695 return false;
696 }
697 });
698
699 if (features.menu.items != null) {
700 HideSelectedAdapter<EventLabel> adapter
701 = new HideSelectedAdapter<EventLabel>(
702 cordova.getActivity(),
703 android.R.layout.simple_spinner_item,
704 features.menu.items);
705 adapter.setDropDownViewResource(
706 android.R.layout.simple_spinner_dropdown_item);
707 menu.setAdapter(adapter);
708 menu.setOnItemSelectedListener(
709 new AdapterView.OnItemSelectedListener() {
710 @Override
711 public void onItemSelected(
712 AdapterView<?> adapterView,
713 View view, int i, long l) {
714 if (inAppWebView != null
715 && i < features.menu.items.length) {
716 emitButtonEvent(
717 features.menu.items[i],
718 inAppWebView.getUrl(), i);
719 }
720 }
721
722 @Override
723 public void onNothingSelected(
724 AdapterView<?> adapterView) {
725 }
726 }
727 );
728 }
729 }
730
731 // Title
732 final TextView title = features.title != null
733 ? new TextView(cordova.getActivity()) : null;
734 if (title != null) {
735 FrameLayout.LayoutParams titleParams
736 = new FrameLayout.LayoutParams(
737 LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
738 titleParams.gravity = Gravity.CENTER;
739 title.setLayoutParams(titleParams);
740 title.setSingleLine();
741 title.setEllipsize(TextUtils.TruncateAt.END);
742 title.setGravity(Gravity.CENTER);
743 title.setTextColor(hexStringToColor(
744 features.title.color != null
745 ? features.title.color : "#000000ff"));
746 if (features.title.staticText != null) {
747 title.setText(features.title.staticText);
748 }
749 }
750
751 // WebView
752 inAppWebView = new WebView(cordova.getActivity());
753 final ViewGroup.LayoutParams inAppWebViewParams = features.fullscreen
754 ? new FrameLayout.LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT)
755 : new LinearLayout.LayoutParams(LayoutParams.MATCH_PARENT, 0);
756 if (!features.fullscreen) {
757 ((LinearLayout.LayoutParams) inAppWebViewParams).weight = 1;
758 }
759 inAppWebView.setLayoutParams(inAppWebViewParams);
760 inAppWebView.setWebChromeClient(new InAppChromeClient(thatWebView));
761 WebViewClient client = new ThemeableBrowserClient(thatWebView, new PageLoadListener() {
762 @Override
763 public void onPageFinished(String url, boolean canGoBack, boolean canGoForward) {
764 if (inAppWebView != null
765 && title != null && features.title != null
766 && features.title.staticText == null
767 && features.title.showPageTitle) {
768 title.setText(inAppWebView.getTitle());
769 }
770
771 if (back != null) {
772 back.setEnabled(canGoBack || features.backButtonCanClose);
773 }
774
775 if (forward != null) {
776 forward.setEnabled(canGoForward);
777 }
778 }
779 });
780 inAppWebView.setWebViewClient(client);
781 WebSettings settings = inAppWebView.getSettings();
782 settings.setJavaScriptEnabled(true);
783 settings.setJavaScriptCanOpenWindowsAutomatically(true);
784 settings.setBuiltInZoomControls(features.zoom);
785 settings.setDisplayZoomControls(false);
786 settings.setPluginState(android.webkit.WebSettings.PluginState.ON);
787
788 //Toggle whether this is enabled or not!
789 Bundle appSettings = cordova.getActivity().getIntent().getExtras();
790 boolean enableDatabase = appSettings == null || appSettings.getBoolean("ThemeableBrowserStorageEnabled", true);
791 if (enableDatabase) {
792 String databasePath = cordova.getActivity().getApplicationContext().getDir("themeableBrowserDB", Context.MODE_PRIVATE).getPath();
793 settings.setDatabasePath(databasePath);
794 settings.setDatabaseEnabled(true);
795 }
796 settings.setDomStorageEnabled(true);
797
798 if (features.clearcache) {
799 CookieManager.getInstance().removeAllCookie();
800 } else if (features.clearsessioncache) {
801 CookieManager.getInstance().removeSessionCookie();
802 }
803
804 inAppWebView.loadUrl(url);
805 inAppWebView.getSettings().setLoadWithOverviewMode(true);
806 inAppWebView.getSettings().setUseWideViewPort(true);
807 inAppWebView.requestFocus();
808 inAppWebView.requestFocusFromTouch();
809
810 // Add buttons to either leftButtonsContainer or
811 // rightButtonsContainer according to user's alignment
812 // configuration.
813 int leftContainerWidth = 0;
814 int rightContainerWidth = 0;
815
816 if (features.customButtons != null) {
817 for (int i = 0; i < features.customButtons.length; i++) {
818 final BrowserButton buttonProps = features.customButtons[i];
819 final int index = i;
820 Button button = createButton(
821 buttonProps,
822 String.format("custom button at %d", i),
823 new View.OnClickListener() {
824 @Override
825 public void onClick(View view) {
826 if (inAppWebView != null) {
827 emitButtonEvent(buttonProps,
828 inAppWebView.getUrl(), index);
829 }
830 }
831 }
832 );
833
834 if (ALIGN_RIGHT.equals(buttonProps.align)) {
835 rightButtonContainer.addView(button);
836 rightContainerWidth
837 += button.getLayoutParams().width;
838 } else {
839 leftButtonContainer.addView(button, 0);
840 leftContainerWidth
841 += button.getLayoutParams().width;
842 }
843 }
844 }
845
846 // Back and forward buttons must be added with special ordering logic such
847 // that back button is always on the left of forward button if both buttons
848 // are on the same side.
849 if (forward != null && features.forwardButton != null
850 && !ALIGN_RIGHT.equals(features.forwardButton.align)) {
851 leftButtonContainer.addView(forward, 0);
852 leftContainerWidth
853 += forward.getLayoutParams().width;
854 }
855
856 if (back != null && features.backButton != null
857 && ALIGN_RIGHT.equals(features.backButton.align)) {
858 rightButtonContainer.addView(back);
859 rightContainerWidth
860 += back.getLayoutParams().width;
861 }
862
863 if (back != null && features.backButton != null
864 && !ALIGN_RIGHT.equals(features.backButton.align)) {
865 leftButtonContainer.addView(back, 0);
866 leftContainerWidth
867 += back.getLayoutParams().width;
868 }
869
870 if (forward != null && features.forwardButton != null
871 && ALIGN_RIGHT.equals(features.forwardButton.align)) {
872 rightButtonContainer.addView(forward);
873 rightContainerWidth
874 += forward.getLayoutParams().width;
875 }
876
877 if (menu != null) {
878 if (features.menu != null
879 && ALIGN_RIGHT.equals(features.menu.align)) {
880 rightButtonContainer.addView(menu);
881 rightContainerWidth
882 += menu.getLayoutParams().width;
883 } else {
884 leftButtonContainer.addView(menu, 0);
885 leftContainerWidth
886 += menu.getLayoutParams().width;
887 }
888 }
889
890 if (close != null) {
891 if (features.closeButton != null
892 && ALIGN_RIGHT.equals(features.closeButton.align)) {
893 rightButtonContainer.addView(close);
894 rightContainerWidth
895 += close.getLayoutParams().width;
896 } else {
897 leftButtonContainer.addView(close, 0);
898 leftContainerWidth
899 += close.getLayoutParams().width;
900 }
901 }
902
903 // Add the views to our toolbar
904 toolbar.addView(leftButtonContainer);
905 // Don't show address bar.
906 // toolbar.addView(edittext);
907 toolbar.addView(rightButtonContainer);
908
909 if (title != null) {
910 int titleMargin = Math.max(
911 leftContainerWidth, rightContainerWidth);
912
913 FrameLayout.LayoutParams titleParams
914 = (FrameLayout.LayoutParams) title.getLayoutParams();
915 titleParams.setMargins(titleMargin, 0, titleMargin, 0);
916 toolbar.addView(title);
917 }
918
919 if (features.fullscreen) {
920 // If full screen mode, we have to add inAppWebView before adding toolbar.
921 main.addView(inAppWebView);
922 }
923
924 // Don't add the toolbar if its been disabled
925 if (features.location) {
926 // Add our toolbar to our main view/layout
927 main.addView(toolbar);
928 }
929
930 if (!features.fullscreen) {
931 // If not full screen, we add inAppWebView after adding toolbar.
932 main.addView(inAppWebView);
933 }
934
935 WindowManager.LayoutParams lp = new WindowManager.LayoutParams();
936 lp.copyFrom(dialog.getWindow().getAttributes());
937 lp.width = WindowManager.LayoutParams.MATCH_PARENT;
938 lp.height = WindowManager.LayoutParams.MATCH_PARENT;
939
940 dialog.setContentView(main);
941 dialog.show();
942 dialog.getWindow().setAttributes(lp);
943 // the goal of openhidden is to load the url and not display it
944 // Show() needs to be called to cause the URL to be loaded
945 if(features.hidden) {
946 dialog.hide();
947 }
948 }
949 };
950 this.cordova.getActivity().runOnUiThread(runnable);
951 return "";
952 }
953
954 /**
955 * Convert our DIP units to Pixels
956 *
957 * @return int
958 */
959 private int dpToPixels(int dipValue) {
960 int value = (int) TypedValue.applyDimension(
961 TypedValue.COMPLEX_UNIT_DIP,
962 (float) dipValue,
963 cordova.getActivity().getResources().getDisplayMetrics()
964 );
965
966 return value;
967 }
968
969 private int hexStringToColor(String hex) {
970 int result = 0;
971
972 if (hex != null && !hex.isEmpty()) {
973 if (hex.charAt(0) == '#') {
974 hex = hex.substring(1);
975 }
976
977 // No alpha, that's fine, we will just attach ff.
978 if (hex.length() < 8) {
979 hex += "ff";
980 }
981
982 result = (int) Long.parseLong(hex, 16);
983
984 // Almost done, but Android color code is in form of ARGB instead of
985 // RGBA, so we gotta shift it a bit.
986 int alpha = (result & 0xff) << 24;
987 result = result >> 8 & 0xffffff | alpha;
988 }
989
990 return result;
991 }
992
993 /**
994 * This is a rather unintuitive helper method to load images. The reason why this method exists
995 * is because due to some service limitations, one may not be able to add images to native
996 * resource bundle. So this method offers a way to load image from www contents instead.
997 * However loading from native resource bundle is already preferred over loading from www. So
998 * if name is given, then it simply loads from resource bundle and the other two parameters are
999 * ignored. If name is not given, then altPath is assumed to be a file path _under_ www and
1000 * altDensity is the desired density of the given image file, because without native resource
1001 * bundle, we can't tell what density the image is supposed to be so it needs to be given
1002 * explicitly.
1003 */
1004 private Drawable getImage(String name, String altPath, double altDensity) throws IOException {
1005 Drawable result = null;
1006 Resources activityRes = cordova.getActivity().getResources();
1007
1008 if (name != null) {
1009 int id = activityRes.getIdentifier(name, "drawable",
1010 cordova.getActivity().getPackageName());
1011 if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
1012 result = activityRes.getDrawable(id);
1013 } else {
1014 result = activityRes.getDrawable(id, cordova.getActivity().getTheme());
1015 }
1016 } else if (altPath != null) {
1017 File file = new File("www", altPath);
1018 InputStream is = null;
1019 try {
1020 is = cordova.getActivity().getAssets().open(file.getPath());
1021 Bitmap bitmap = BitmapFactory.decodeStream(is);
1022 bitmap.setDensity((int) (DisplayMetrics.DENSITY_MEDIUM * altDensity));
1023 result = new BitmapDrawable(activityRes, bitmap);
1024 } finally {
1025 // Make sure we close this input stream to prevent resource leak.
1026 try {
1027 is.close();
1028 } catch (Exception e) {}
1029 }
1030 }
1031 return result;
1032 }
1033
1034 private void setButtonImages(View view, BrowserButton buttonProps, int disabledAlpha) {
1035 Drawable normalDrawable = null;
1036 Drawable disabledDrawable = null;
1037 Drawable pressedDrawable = null;
1038
1039 CharSequence description = view.getContentDescription();
1040
1041 if (buttonProps.image != null || buttonProps.wwwImage != null) {
1042 try {
1043 normalDrawable = getImage(buttonProps.image, buttonProps.wwwImage,
1044 buttonProps.wwwImageDensity);
1045 ViewGroup.LayoutParams params = view.getLayoutParams();
1046 params.width = normalDrawable.getIntrinsicWidth();
1047 params.height = normalDrawable.getIntrinsicHeight();
1048 } catch (Resources.NotFoundException e) {
1049 emitError(ERR_LOADFAIL,
1050 String.format("Image for %s, %s, failed to load",
1051 description, buttonProps.image));
1052 } catch (IOException ioe) {
1053 emitError(ERR_LOADFAIL,
1054 String.format("Image for %s, %s, failed to load",
1055 description, buttonProps.wwwImage));
1056 }
1057 } else {
1058 emitWarning(WRN_UNDEFINED,
1059 String.format("Image for %s is not defined. Button will not be shown",
1060 description));
1061 }
1062
1063 if (buttonProps.imagePressed != null || buttonProps.wwwImagePressed != null) {
1064 try {
1065 pressedDrawable = getImage(buttonProps.imagePressed, buttonProps.wwwImagePressed,
1066 buttonProps.wwwImageDensity);
1067 } catch (Resources.NotFoundException e) {
1068 emitError(ERR_LOADFAIL,
1069 String.format("Pressed image for %s, %s, failed to load",
1070 description, buttonProps.imagePressed));
1071 } catch (IOException e) {
1072 emitError(ERR_LOADFAIL,
1073 String.format("Pressed image for %s, %s, failed to load",
1074 description, buttonProps.wwwImagePressed));
1075 }
1076 } else {
1077 emitWarning(WRN_UNDEFINED,
1078 String.format("Pressed image for %s is not defined.",
1079 description));
1080 }
1081
1082 if (normalDrawable != null) {
1083 // Create the disabled state drawable by fading the normal state
1084 // drawable. Drawable.setAlpha() stopped working above Android 4.4
1085 // so we gotta bring out some bitmap magic. Credit goes to:
1086 // http://stackoverflow.com/a/7477572
1087 Bitmap enabledBitmap = ((BitmapDrawable) normalDrawable).getBitmap();
1088 Bitmap disabledBitmap = Bitmap.createBitmap(
1089 normalDrawable.getIntrinsicWidth(),
1090 normalDrawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
1091 Canvas canvas = new Canvas(disabledBitmap);
1092
1093 Paint paint = new Paint();
1094 paint.setAlpha(disabledAlpha);
1095 canvas.drawBitmap(enabledBitmap, 0, 0, paint);
1096
1097 Resources activityRes = cordova.getActivity().getResources();
1098 disabledDrawable = new BitmapDrawable(activityRes, disabledBitmap);
1099 }
1100
1101 StateListDrawable states = new StateListDrawable();
1102 if (pressedDrawable != null) {
1103 states.addState(
1104 new int[] {
1105 android.R.attr.state_pressed
1106 },
1107 pressedDrawable
1108 );
1109 }
1110 if (normalDrawable != null) {
1111 states.addState(
1112 new int[] {
1113 android.R.attr.state_enabled
1114 },
1115 normalDrawable
1116 );
1117 }
1118 if (disabledDrawable != null) {
1119 states.addState(
1120 new int[] {},
1121 disabledDrawable
1122 );
1123 }
1124
1125 setBackground(view, states);
1126 }
1127
1128 private void setBackground(View view, Drawable drawable) {
1129 if (Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN) {
1130 view.setBackgroundDrawable(drawable);
1131 } else {
1132 view.setBackground(drawable);
1133 }
1134 }
1135
1136 private Button createButton(BrowserButton buttonProps, String description,
1137 View.OnClickListener listener) {
1138 Button result = null;
1139 if (buttonProps != null) {
1140 result = new Button(cordova.getActivity());
1141 result.setContentDescription(description);
1142 result.setLayoutParams(new LinearLayout.LayoutParams(
1143 LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));
1144 setButtonImages(result, buttonProps, DISABLED_ALPHA);
1145 if (listener != null) {
1146 result.setOnClickListener(listener);
1147 }
1148 } else {
1149 emitWarning(WRN_UNDEFINED,
1150 String.format("%s is not defined. Button will not be shown.",
1151 description));
1152 }
1153 return result;
1154 }
1155
1156 /**
1157 * Create a new plugin success result and send it back to JavaScript
1158 *
1159 * @param obj a JSONObject contain event payload information
1160 */
1161 private void sendUpdate(JSONObject obj, boolean keepCallback) {
1162 sendUpdate(obj, keepCallback, PluginResult.Status.OK);
1163 }
1164
1165 /**
1166 * Create a new plugin result and send it back to JavaScript
1167 *
1168 * @param obj a JSONObject contain event payload information
1169 * @param status the status code to return to the JavaScript environment
1170 */
1171 private void sendUpdate(JSONObject obj, boolean keepCallback, PluginResult.Status status) {
1172 if (callbackContext != null) {
1173 PluginResult result = new PluginResult(status, obj);
1174 result.setKeepCallback(keepCallback);
1175 callbackContext.sendPluginResult(result);
1176 if (!keepCallback) {
1177 callbackContext = null;
1178 }
1179 }
1180 }
1181
1182 public static interface PageLoadListener {
1183 public void onPageFinished(String url, boolean canGoBack,
1184 boolean canGoForward);
1185 }
1186
1187 /**
1188 * The webview client receives notifications about appView
1189 */
1190 public class ThemeableBrowserClient extends WebViewClient {
1191 PageLoadListener callback;
1192 CordovaWebView webView;
1193
1194 /**
1195 * Constructor.
1196 *
1197 * @param webView
1198 * @param callback
1199 */
1200 public ThemeableBrowserClient(CordovaWebView webView,
1201 PageLoadListener callback) {
1202 this.webView = webView;
1203 this.callback = callback;
1204 }
1205
1206 /**
1207 * Override the URL that should be loaded
1208 *
1209 * This handles a small subset of all the URIs that would be encountered.
1210 *
1211 * @param webView
1212 * @param url
1213 */
1214 @Override
1215 public boolean shouldOverrideUrlLoading(WebView webView, String url) {
1216 if (url.startsWith(WebView.SCHEME_TEL)) {
1217 try {
1218 Intent intent = new Intent(Intent.ACTION_DIAL);
1219 intent.setData(Uri.parse(url));
1220 cordova.getActivity().startActivity(intent);
1221 return true;
1222 } catch (android.content.ActivityNotFoundException e) {
1223 Log.e(LOG_TAG, "Error dialing " + url + ": " + e.toString());
1224 }
1225 } else if (url.startsWith("geo:") || url.startsWith(WebView.SCHEME_MAILTO) || url.startsWith("market:")) {
1226 try {
1227 Intent intent = new Intent(Intent.ACTION_VIEW);
1228 intent.setData(Uri.parse(url));
1229 cordova.getActivity().startActivity(intent);
1230 return true;
1231 } catch (android.content.ActivityNotFoundException e) {
1232 Log.e(LOG_TAG, "Error with " + url + ": " + e.toString());
1233 }
1234 }
1235 // If sms:5551212?body=This is the message
1236 else if (url.startsWith("sms:")) {
1237 try {
1238 Intent intent = new Intent(Intent.ACTION_VIEW);
1239
1240 // Get address
1241 String address = null;
1242 int parmIndex = url.indexOf('?');
1243 if (parmIndex == -1) {
1244 address = url.substring(4);
1245 } else {
1246 address = url.substring(4, parmIndex);
1247
1248 // If body, then set sms body
1249 Uri uri = Uri.parse(url);
1250 String query = uri.getQuery();
1251 if (query != null) {
1252 if (query.startsWith("body=")) {
1253 intent.putExtra("sms_body", query.substring(5));
1254 }
1255 }
1256 }
1257 intent.setData(Uri.parse("sms:" + address));
1258 intent.putExtra("address", address);
1259 intent.setType("vnd.android-dir/mms-sms");
1260 cordova.getActivity().startActivity(intent);
1261 return true;
1262 } catch (android.content.ActivityNotFoundException e) {
1263 Log.e(LOG_TAG, "Error sending sms " + url + ":" + e.toString());
1264 }
1265 }
1266 return false;
1267 }
1268
1269
1270 /*
1271 * onPageStarted fires the LOAD_START_EVENT
1272 *
1273 * @param view
1274 * @param url
1275 * @param favicon
1276 */
1277 @Override
1278 public void onPageStarted(WebView view, String url, Bitmap favicon) {
1279 super.onPageStarted(view, url, favicon);
1280 String newloc = "";
1281 if (url.startsWith("http:") || url.startsWith("https:") || url.startsWith("file:")) {
1282 newloc = url;
1283 }
1284 else
1285 {
1286 // Assume that everything is HTTP at this point, because if we don't specify,
1287 // it really should be. Complain loudly about this!!!
1288 Log.e(LOG_TAG, "Possible Uncaught/Unknown URI");
1289 newloc = "http://" + url;
1290 }
1291
1292 // Update the UI if we haven't already
1293 if (!newloc.equals(edittext.getText().toString())) {
1294 edittext.setText(newloc);
1295 }
1296
1297 try {
1298 JSONObject obj = new JSONObject();
1299 obj.put("type", LOAD_START_EVENT);
1300 obj.put("url", newloc);
1301 sendUpdate(obj, true);
1302 } catch (JSONException ex) {
1303 Log.e(LOG_TAG, "URI passed in has caused a JSON error.");
1304 }
1305 }
1306
1307 public void onPageFinished(WebView view, String url) {
1308 super.onPageFinished(view, url);
1309
1310 try {
1311 JSONObject obj = new JSONObject();
1312 obj.put("type", LOAD_STOP_EVENT);
1313 obj.put("url", url);
1314
1315 sendUpdate(obj, true);
1316
1317 if (this.callback != null) {
1318 this.callback.onPageFinished(url, view.canGoBack(),
1319 view.canGoForward());
1320 }
1321 } catch (JSONException ex) {
1322 }
1323 }
1324
1325 public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
1326 super.onReceivedError(view, errorCode, description, failingUrl);
1327
1328 try {
1329 JSONObject obj = new JSONObject();
1330 obj.put("type", LOAD_ERROR_EVENT);
1331 obj.put("url", failingUrl);
1332 obj.put("code", errorCode);
1333 obj.put("message", description);
1334
1335 sendUpdate(obj, true, PluginResult.Status.ERROR);
1336 } catch (JSONException ex) {
1337 }
1338 }
1339 }
1340
1341 /**
1342 * Like Spinner but will always trigger onItemSelected even if a selected
1343 * item is selected, and always ignore default selection.
1344 */
1345 public class MenuSpinner extends Spinner {
1346 private OnItemSelectedListener listener;
1347
1348 public MenuSpinner(Context context) {
1349 super(context);
1350 }
1351
1352 @Override
1353 public void setSelection(int position) {
1354 super.setSelection(position);
1355
1356 if (listener != null) {
1357 listener.onItemSelected(null, this, position, 0);
1358 }
1359 }
1360
1361 @Override
1362 public void setOnItemSelectedListener(OnItemSelectedListener listener) {
1363 this.listener = listener;
1364 }
1365 }
1366
1367 /**
1368 * Extension of ArrayAdapter. The only difference is that it hides the
1369 * selected text that's shown inside spinner.
1370 * @param <T>
1371 */
1372 private static class HideSelectedAdapter<T> extends ArrayAdapter {
1373
1374 public HideSelectedAdapter(Context context, int resource, T[] objects) {
1375 super(context, resource, objects);
1376 }
1377
1378 public View getView (int position, View convertView, ViewGroup parent) {
1379 View v = super.getView(position, convertView, parent);
1380 v.setVisibility(View.GONE);
1381 return v;
1382 }
1383 }
1384
1385
1386 /**
1387 * A class to hold parsed option properties.
1388 */
1389 private static class Options {
1390 public boolean location = true;
1391 public boolean hidden = false;
1392 public boolean clearcache = false;
1393 public boolean clearsessioncache = false;
1394 public boolean zoom = true;
1395 public boolean hardwareback = true;
1396
1397 public Toolbar toolbar;
1398 public Title title;
1399 public BrowserButton backButton;
1400
在搞之前 首先要有常识 在 Android 手机中内置了一款高性能 webkit 内核浏览器,在 SDK 中封装为一个叫做 WebView 组件.
其次 在打开页面时选择横竖屏 , 然后去谷歌搜下 android webview 横竖屏 比如下面这篇文章截图 知道 Activity类可以通过setRequestedOrientation 方法设置 横竖屏显示
我们调用js 是用的 ThemeableBrowser 类的open 看上面的代码copy
发现它重写了父类 CordovaPlugin 的execute方法去执行open操作 我们调用的方法参数是
js调用第二个参数是 _black 所以直接看188行 然后跳转526行 showWebPage
看752行
inAppWebView = new WebView(cordova.getActivity());
804行
inAppWebView.loadUrl(url);
loadUrl就是把webview展示出来, 这个webview实例是通过 752行 cordova.getActivity()
cordova.getActivity()
然后搜索 cordova.getActivity() 发现 cordova是个接口 然后搜一下这个接口的实现类 发现是 CordovaInterfaceImpl这个类 发现他的getActivity方法是返回的 构造函数注入的 activity
然后再找 CordovaInterfaceImpl 实在 CordovaActivity.java的 makeCordovaInterface 方法中实例化的
makeCordovaInterface 是在 CordovaActivity的 oncreate方法实例化的
CordovaActivity 继承 Activity MainActivity又继承CordovaActivity
Activity是Android的四大组件之一。是用户操作的可视化界面 MainActivity一般作为主页面展示
Android手机APP启动的第一个Activity是可以自己设置的,不是必须的MainActivity,可以是任何的Activity。
设置Android手机APP启动的第一个Activity得看Android项目里的mainfest.xml文件:
简单理一下就是
app启动 => MainActivity初始化 onCreate方法 => makeCordovaInterface(this) 把MainActivity本身最为参数实例化CordovaInterfaceImpl类 => CordovaInterface作为接口参数cordova 被组合到 ThemeableBrowser类
js调用 ThemeableBrowser.open 方法实例化一个inAppWebView = new WebView(cordova.getActivity());
我们通过给MainActivity设置横竖屏的显示方式( setRequestedOrientation) 然后展示 url页面 inAppWebView.loadUrl(url);
再看关闭按钮事件 671行 调用 closeDialog
我们在这里用 setRequestedOrientation把 Activity改为竖屏
整个解决方式是
js调用时候