Pebble: refactor the background webview

Now native controls seem to work (including datepicker), still the webview is not started upon watchapp start, but when long-pressing each app in the app manager. After the webview is started it will live in the background until device disconnect.
This commit is contained in:
Daniele Gobbetti 2017-01-01 18:33:39 +01:00
parent 6d02a76328
commit eaaa940637
4 changed files with 103 additions and 102 deletions

View File

@ -7,6 +7,7 @@ import android.os.Build;
import android.os.Bundle; import android.os.Bundle;
import android.support.v4.app.NavUtils; import android.support.v4.app.NavUtils;
import android.view.MenuItem; import android.view.MenuItem;
import android.view.View;
import android.webkit.JavascriptInterface; import android.webkit.JavascriptInterface;
import android.webkit.ValueCallback; import android.webkit.ValueCallback;
import android.webkit.WebView; import android.webkit.WebView;
@ -45,39 +46,30 @@ public class ExternalPebbleJSActivity extends GBActivity {
throw new IllegalArgumentException("Must provide a device when invoking this activity"); throw new IllegalArgumentException("Must provide a device when invoking this activity");
} }
setContentView(R.layout.activity_external_pebble_js); setContentView(R.layout.activity_external_pebble_js);
WebViewSingleton.getorInitWebView(this, mGBDevice, appUuid);
}
@Override
protected void onNewIntent(Intent incoming) {
incoming.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
super.onNewIntent(incoming);
confUri = incoming.getData();
}
@Override
protected void onResume() {
super.onResume();
String queryString = "";
myWebView = WebViewSingleton.getorInitWebView(this, mGBDevice, appUuid); myWebView = WebViewSingleton.getorInitWebView(this, mGBDevice, appUuid);
myWebView.addJavascriptInterface(new ActivityJSInterface(this), "GBActivity"); myWebView.setWillNotDraw(false);
myWebView.addJavascriptInterface(new ActivityJSInterface(ExternalPebbleJSActivity.this), "GBActivity");
FrameLayout fl = (FrameLayout) findViewById(R.id.webview_placeholder); FrameLayout fl = (FrameLayout) findViewById(R.id.webview_placeholder);
if (myWebView != null)
fl.addView(myWebView); fl.addView(myWebView);
//needed to display clay dialogs
myWebView.getSettings().setUseWideViewPort(true); myWebView.addOnAttachStateChangeListener(new View.OnAttachStateChangeListener() {
myWebView.requestFocus(); @Override
public void onViewAttachedToWindow(View v) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
// chromium, enable hardware acceleration
v.setLayerType(View.LAYER_TYPE_HARDWARE, null);
} else {
// older android version, disable hardware acceleration
v.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
}
String queryString = "";
if (confUri != null) { if (confUri != null) {
//getting back with configuration data //getting back with configuration data
try { try {
@ -86,7 +78,7 @@ public class ExternalPebbleJSActivity extends GBActivity {
} catch (IllegalArgumentException e) { } catch (IllegalArgumentException e) {
GB.toast("returned uri: " + confUri.toString(), Toast.LENGTH_LONG, GB.ERROR); GB.toast("returned uri: " + confUri.toString(), Toast.LENGTH_LONG, GB.ERROR);
} }
myWebView.loadUrl("file:///android_asset/app_config/configure.html?" + queryString); ((WebView) v).loadUrl("file:///android_asset/app_config/configure.html?" + queryString);
} else { } else {
//show configuration //show configuration
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
@ -97,18 +89,29 @@ public class ExternalPebbleJSActivity extends GBActivity {
} }
}); });
} else { } else {
myWebView.loadUrl("javascript:Pebble.evaluate('showConfiguration');"); ((WebView) v).loadUrl("javascript:Pebble.evaluate('showConfiguration');");
} }
} }
} }
@Override @Override
protected void onPause() { public void onViewDetachedFromWindow(View v) {
myWebView.removeJavascriptInterface("GBActivity");
myWebView.setWillNotDraw(true);
FrameLayout fl = (FrameLayout) findViewById(R.id.webview_placeholder); FrameLayout fl = (FrameLayout) findViewById(R.id.webview_placeholder);
fl.removeAllViews(); fl.removeAllViews();
super.onPause(); }
});
}
@Override
protected void onNewIntent(Intent incoming) {
incoming.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
super.onNewIntent(incoming);
confUri = incoming.getData();
} }
@Override @Override

View File

@ -39,6 +39,7 @@ import nodomain.freeyourgadget.gadgetbridge.model.DeviceService;
import nodomain.freeyourgadget.gadgetbridge.service.devices.pebble.PebbleProtocol; import nodomain.freeyourgadget.gadgetbridge.service.devices.pebble.PebbleProtocol;
import nodomain.freeyourgadget.gadgetbridge.util.FileUtils; import nodomain.freeyourgadget.gadgetbridge.util.FileUtils;
import nodomain.freeyourgadget.gadgetbridge.util.PebbleUtils; import nodomain.freeyourgadget.gadgetbridge.util.PebbleUtils;
import nodomain.freeyourgadget.gadgetbridge.util.WebViewSingleton;
public abstract class AbstractAppManagerFragment extends Fragment { public abstract class AbstractAppManagerFragment extends Fragment {
@ -323,6 +324,8 @@ public abstract class AbstractAppManagerFragment extends Fragment {
view.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS); view.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
popupMenu.show(); popupMenu.show();
//TODO: replace with local broadcast on app start
WebViewSingleton.getorInitWebView(getActivity(), mGBDevice, selectedApp.getUUID());
return true; return true;
} }

View File

@ -570,6 +570,7 @@ class PebbleIoThread extends GBDeviceIoThread {
case START: case START:
LOG.info("got GBDeviceEventAppManagement START event for uuid: " + appMgmt.uuid); LOG.info("got GBDeviceEventAppManagement START event for uuid: " + appMgmt.uuid);
WebViewSingleton.getorInitWebView(getContext(), gbDevice, appMgmt.uuid); WebViewSingleton.getorInitWebView(getContext(), gbDevice, appMgmt.uuid);
//TODO: the method call above will not work the first time as we need an activity. Either we find a way to have one here, or replace it with a local broadcast
break; break;
default: default:
break; break;

View File

@ -1,12 +1,10 @@
package nodomain.freeyourgadget.gadgetbridge.util; package nodomain.freeyourgadget.gadgetbridge.util;
import android.app.Activity;
import android.content.Context; import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.content.MutableContextWrapper; import android.content.MutableContextWrapper;
import android.net.Uri; import android.net.Uri;
import android.os.Handler;
import android.os.Looper;
import android.util.Log;
import android.webkit.ConsoleMessage; import android.webkit.ConsoleMessage;
import android.webkit.JavascriptInterface; import android.webkit.JavascriptInterface;
import android.webkit.WebChromeClient; import android.webkit.WebChromeClient;
@ -35,7 +33,7 @@ import java.util.UUID;
import nodomain.freeyourgadget.gadgetbridge.GBApplication; import nodomain.freeyourgadget.gadgetbridge.GBApplication;
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice; import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
public class WebViewSingleton { public class WebViewSingleton extends Activity {
private static final Logger LOG = LoggerFactory.getLogger(WebViewSingleton.class); private static final Logger LOG = LoggerFactory.getLogger(WebViewSingleton.class);
@ -46,19 +44,20 @@ public class WebViewSingleton {
private WebViewSingleton() { private WebViewSingleton() {
} }
public static WebView getorInitWebView(final Context context, final GBDevice device, final UUID uuid) { public static WebView getorInitWebView(Context context, GBDevice device, UUID uuid) {
if (context instanceof Activity) {
final MutableContextWrapper _contextWrapper = new MutableContextWrapper(context); final MutableContextWrapper _contextWrapper = new MutableContextWrapper(context);
if (contextWrapper != null) { if (contextWrapper != null) {
contextWrapper.setBaseContext(context); contextWrapper.setBaseContext(context);
} else { } else {
contextWrapper = _contextWrapper; contextWrapper = _contextWrapper;
} }
}
new Handler(Looper.getMainLooper()).post(new Runnable() { //here we are sure that contextWrapper is either null or an activity, hence we run on the main thread
@Override if (contextWrapper != null) {
public void run() {
if (instance == null) { if (instance == null) {
instance = new WebView(_contextWrapper); instance = new WebView(contextWrapper);
instance.setWillNotDraw(true);
instance.clearCache(true); instance.clearCache(true);
instance.setWebViewClient(new GBWebClient()); instance.setWebViewClient(new GBWebClient());
instance.setWebChromeClient(new GBChromeClient()); instance.setWebChromeClient(new GBChromeClient());
@ -79,24 +78,19 @@ public class WebViewSingleton {
} else { } else {
LOG.debug("Not reloading the webview " + jsInterface.mUuid.toString()); LOG.debug("Not reloading the webview " + jsInterface.mUuid.toString());
} }
} else {
LOG.debug("WEBV: not using the passed context, as it is not an activity");
} }
});
return instance; return instance;
} }
public static void disposeWebView() { public static void disposeWebView() {
new Handler(Looper.getMainLooper()).post(new Runnable() {
@Override
public void run() {
if (instance != null) { if (instance != null) {
instance.destroy(); instance.destroy();
instance = null; instance = null;
} }
} }
});
}
private static class GBChromeClient extends WebChromeClient { private static class GBChromeClient extends WebChromeClient {
@Override @Override
@ -155,7 +149,7 @@ public class WebViewSingleton {
@JavascriptInterface @JavascriptInterface
public void gbLog(String msg) { public void gbLog(String msg) {
Log.d("WEBVIEW", msg); LOG.debug("WEBVIEW", msg);
} }
@JavascriptInterface @JavascriptInterface
@ -196,7 +190,7 @@ public class WebViewSingleton {
} }
} }
LOG.info(out.toString()); LOG.info("WEBV:" + out.toString());
GBApplication.deviceService().onAppConfiguration(this.mUuid, out.toString()); GBApplication.deviceService().onAppConfiguration(this.mUuid, out.toString());
} catch (JSONException e) { } catch (JSONException e) {