1 // Copyright (c) 2013 Intel Corporation. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 package org.xwalk.core.xwview.shell;
7 import android.app.Activity;
8 import android.content.BroadcastReceiver;
9 import android.content.Intent;
10 import android.content.Context;
11 import android.content.IntentFilter;
12 import android.graphics.drawable.ClipDrawable;
13 import android.os.Bundle;
14 import android.os.Looper;
15 import android.os.MessageQueue;
16 import android.util.Log;
17 import android.view.KeyEvent;
18 import android.view.View;
19 import android.view.View.OnClickListener;
20 import android.view.View.OnFocusChangeListener;
21 import android.view.inputmethod.EditorInfo;
22 import android.view.inputmethod.InputMethodManager;
23 import android.widget.EditText;
24 import android.widget.ImageButton;
25 import android.widget.LinearLayout;
26 import android.widget.TextView;
27 import android.widget.TextView.OnEditorActionListener;
29 import org.chromium.content.app.LibraryLoader;
30 import org.chromium.content.browser.TracingControllerAndroid;
31 import org.chromium.content.common.CommandLine;
32 import org.xwalk.core.client.XWalkDefaultWebChromeClient;
33 import org.xwalk.core.XWalkView;
35 public class XWalkViewShellActivity extends Activity {
36 public static final String COMMAND_LINE_FILE = "/data/local/tmp/xwview-shell-command-line";
37 private static final String TAG = XWalkViewShellActivity.class.getName();
38 public static final String COMMAND_LINE_ARGS_KEY = "commandLineArgs";
39 private static final long COMPLETED_PROGRESS_TIMEOUT_MS = 200;
40 private static final String ACTION_LAUNCH_URL = "org.xwalk.core.xwview.shell.launch";
42 private LinearLayout mToolbar;
43 private EditText mUrlTextView;
44 private ImageButton mPrevButton;
45 private ImageButton mNextButton;
46 private ImageButton mStopButton;
47 private ImageButton mReloadButton;
48 private ClipDrawable mProgressDrawable;
49 private XWalkView mView;
50 private TracingControllerAndroid mTracingController;
51 private BroadcastReceiver mReceiver;
53 private Runnable mClearProgressRunnable = new Runnable() {
56 mProgressDrawable.setLevel(0);
60 TracingControllerAndroid getTracingController() {
61 if (mTracingController == null) {
62 mTracingController = new TracingControllerAndroid(this);
64 return mTracingController;
67 private void registerTracingReceiverWhenIdle() {
68 // Delay tracing receiver registration until the main loop is idle.
69 Looper.myQueue().addIdleHandler(new MessageQueue.IdleHandler() {
71 public boolean queueIdle() {
72 // Will retry if the native library is not initialized yet.
73 if (!LibraryLoader.isInitialized()) return true;
75 getTracingController().registerReceiver(XWalkViewShellActivity.this);
76 } catch (SecurityException e) {
77 Log.w(TAG, "failed to register tracing receiver: " + e.getMessage());
84 private void unregisterTracingReceiver() {
86 getTracingController().unregisterReceiver(this);
87 } catch (SecurityException e) {
88 Log.w(TAG, "failed to unregister tracing receiver: " + e.getMessage());
93 public void onCreate(Bundle savedInstanceState) {
94 super.onCreate(savedInstanceState);
96 registerTracingReceiverWhenIdle();
98 if (!CommandLine.isInitialized()) {
99 CommandLine.initFromFile(COMMAND_LINE_FILE);
100 String[] commandLineParams = getCommandLineParamsFromIntent(getIntent());
101 if (commandLineParams != null) {
102 CommandLine.getInstance().appendSwitchesAndArguments(commandLineParams);
106 waitForDebuggerIfNeeded();
108 setContentView(R.layout.testshell_activity);
110 mView = (XWalkView) findViewById(R.id.content_container);
111 mToolbar = (LinearLayout) findViewById(R.id.toolbar);
112 mProgressDrawable = (ClipDrawable) findViewById(R.id.toolbar).getBackground();
114 initializeUrlField();
116 initializeXWalkViewClients();
118 IntentFilter intentFilter = new IntentFilter(ACTION_LAUNCH_URL);
119 mReceiver = new BroadcastReceiver() {
121 public void onReceive(Context context, Intent intent) {
122 Bundle bundle = intent.getExtras();
126 if (bundle.containsKey("url")) {
127 String extra = bundle.getString("url");
128 mView.loadUrl(sanitizeUrl(extra));
132 registerReceiver(mReceiver, intentFilter);
133 mView.enableRemoteDebugging();
137 public void onPause() {
143 public void onResume() {
149 public void onDestroy() {
151 unregisterReceiver(mReceiver);
152 unregisterTracingReceiver();
157 protected void onActivityResult(int requestCode, int resultCode, Intent data) {
158 mView.onActivityResult(requestCode, resultCode, data);
162 public boolean onKeyUp(int keyCode, KeyEvent event) {
163 return mView.onKeyUp(keyCode, event) || super.onKeyUp(keyCode, event);
166 private void waitForDebuggerIfNeeded() {
167 if (CommandLine.getInstance().hasSwitch(CommandLine.WAIT_FOR_JAVA_DEBUGGER)) {
168 Log.e(TAG, "Waiting for Java debugger to connect...");
169 android.os.Debug.waitForDebugger();
170 Log.e(TAG, "Java debugger connected. Resuming execution.");
174 private static String[] getCommandLineParamsFromIntent(Intent intent) {
175 return intent != null ? intent.getStringArrayExtra(COMMAND_LINE_ARGS_KEY) : null;
178 private static String sanitizeUrl(String url) {
179 if (url == null) return url;
180 if (url.startsWith("www.") || url.indexOf(":") == -1) url = "http://" + url;
184 private void initializeUrlField() {
185 mUrlTextView = (EditText) findViewById(R.id.url);
186 mUrlTextView.setOnEditorActionListener(new OnEditorActionListener() {
188 public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
189 if ((actionId != EditorInfo.IME_ACTION_GO) && (event == null ||
190 event.getKeyCode() != KeyEvent.KEYCODE_ENTER ||
191 event.getAction() != KeyEvent.ACTION_DOWN)) {
195 mView.loadUrl(sanitizeUrl(mUrlTextView.getText().toString()));
196 mUrlTextView.clearFocus();
197 setKeyboardVisibilityForUrl(false);
201 mUrlTextView.setOnFocusChangeListener(new OnFocusChangeListener() {
203 public void onFocusChange(View v, boolean hasFocus) {
204 setKeyboardVisibilityForUrl(hasFocus);
205 mNextButton.setVisibility(hasFocus ? View.GONE : View.VISIBLE);
206 mPrevButton.setVisibility(hasFocus ? View.GONE : View.VISIBLE);
207 mStopButton.setVisibility(hasFocus ? View.GONE : View.VISIBLE);
208 mReloadButton.setVisibility(hasFocus ? View.GONE : View.VISIBLE);
210 mUrlTextView.setText(mView.getUrl());
217 private void initializeButtons() {
218 mPrevButton = (ImageButton) findViewById(R.id.prev);
219 mPrevButton.setOnClickListener(new OnClickListener() {
221 public void onClick(View v) {
222 if (mView.canGoBack()) mView.goBack();
226 mNextButton = (ImageButton) findViewById(R.id.next);
227 mNextButton.setOnClickListener(new OnClickListener() {
229 public void onClick(View v) {
230 if (mView.canGoForward()) mView.goForward();
234 mStopButton = (ImageButton) findViewById(R.id.stop);
235 mStopButton.setOnClickListener(new OnClickListener() {
237 public void onClick(View v) {
242 mReloadButton = (ImageButton) findViewById(R.id.reload);
243 mReloadButton.setOnClickListener(new OnClickListener() {
245 public void onClick(View v) {
251 private void initializeXWalkViewClients() {
252 mView.setXWalkWebChromeClient(new XWalkDefaultWebChromeClient(this, mView) {
253 public void onProgressChanged(XWalkView view, int newProgress) {
254 mToolbar.removeCallbacks(mClearProgressRunnable);
256 mProgressDrawable.setLevel((int) (100.0 * newProgress));
257 if (newProgress == 100)
258 mToolbar.postDelayed(mClearProgressRunnable, COMPLETED_PROGRESS_TIMEOUT_MS);
259 mUrlTextView.setText(mView.getUrl());
263 private void setKeyboardVisibilityForUrl(boolean visible) {
264 InputMethodManager imm = (InputMethodManager) getSystemService(
265 Context.INPUT_METHOD_SERVICE);
267 imm.showSoftInput(mUrlTextView, InputMethodManager.SHOW_IMPLICIT);
269 imm.hideSoftInputFromWindow(mUrlTextView.getWindowToken(), 0);