1 /* 2 * Copyright (C) 2009 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.browser; 18 19 import android.content.Context; 20 import android.database.DataSetObserver; 21 import android.util.AttributeSet; 22 import android.view.LayoutInflater; 23 import android.view.View; 24 import android.view.ViewGroup; 25 import android.view.View.OnClickListener; 26 import android.webkit.WebView; 27 import android.widget.Button; 28 import android.widget.EditText; 29 import android.widget.LinearLayout; 30 import android.widget.ListView; 31 import android.widget.TextView; 32 import android.widget.TwoLineListItem; 33 34 import java.util.Vector; 35 36 /* package */ class ErrorConsoleView extends LinearLayout { 37 38 /** 39 * Define some constants to describe the visibility of the error console. 40 */ 41 public static final int SHOW_MINIMIZED = 0; 42 public static final int SHOW_MAXIMIZED = 1; 43 public static final int SHOW_NONE = 2; 44 45 private TextView mConsoleHeader; 46 private ErrorConsoleListView mErrorList; 47 private LinearLayout mEvalJsViewGroup; 48 private EditText mEvalEditText; 49 private Button mEvalButton; 50 private WebView mWebView; 51 private int mCurrentShowState = SHOW_NONE; 52 53 private boolean mSetupComplete = false; 54 55 // Before we've been asked to display the console, cache any messages that should 56 // be added to the console. Then when we do display the console, add them to the view 57 // then. 58 private Vector<ErrorConsoleMessage> mErrorMessageCache; 59 ErrorConsoleView(Context context)60 public ErrorConsoleView(Context context) { 61 super(context); 62 } 63 ErrorConsoleView(Context context, AttributeSet attributes)64 public ErrorConsoleView(Context context, AttributeSet attributes) { 65 super(context, attributes); 66 } 67 commonSetupIfNeeded()68 private void commonSetupIfNeeded() { 69 if (mSetupComplete) { 70 return; 71 } 72 73 LayoutInflater inflater = (LayoutInflater) getContext().getSystemService( 74 Context.LAYOUT_INFLATER_SERVICE); 75 inflater.inflate(R.layout.error_console, this); 76 77 // Get references to each ui element. 78 mConsoleHeader = (TextView) findViewById(R.id.error_console_header_id); 79 mErrorList = (ErrorConsoleListView) findViewById(R.id.error_console_list_id); 80 mEvalJsViewGroup = (LinearLayout) findViewById(R.id.error_console_eval_view_group_id); 81 mEvalEditText = (EditText) findViewById(R.id.error_console_eval_text_id); 82 mEvalButton = (Button) findViewById(R.id.error_console_eval_button_id); 83 84 mEvalButton.setOnClickListener(new OnClickListener() { 85 public void onClick(View v) { 86 // Send the javascript to be evaluated to webkit as a javascript: url 87 // TODO: Can we expose access to webkit's JS interpreter here and evaluate it that 88 // way? Note that this is called on the UI thread so we will need to post a message 89 // to the WebCore thread to implement this. 90 if (mWebView != null) { 91 mWebView.loadUrl("javascript:" + mEvalEditText.getText()); 92 } 93 94 mEvalEditText.setText(""); 95 } 96 }); 97 98 // Make clicking on the console title bar min/maximse it. 99 mConsoleHeader.setOnClickListener(new OnClickListener() { 100 public void onClick(View v) { 101 if (mCurrentShowState == SHOW_MINIMIZED) { 102 showConsole(SHOW_MAXIMIZED); 103 } else { 104 showConsole(SHOW_MINIMIZED); 105 } 106 } 107 }); 108 109 // Add any cached messages to the list now that we've assembled the view. 110 if (mErrorMessageCache != null) { 111 for (ErrorConsoleMessage msg : mErrorMessageCache) { 112 mErrorList.addErrorMessage(msg.getMessage(), msg.getSourceID(), msg.getLineNumber()); 113 } 114 mErrorMessageCache.clear(); 115 } 116 117 mSetupComplete = true; 118 } 119 120 /** 121 * Adds a message to the set of messages the console uses. 122 */ addErrorMessage(String msg, String sourceId, int lineNumber)123 public void addErrorMessage(String msg, String sourceId, int lineNumber) { 124 if (mSetupComplete) { 125 mErrorList.addErrorMessage(msg, sourceId, lineNumber); 126 } else { 127 if (mErrorMessageCache == null) { 128 mErrorMessageCache = new Vector<ErrorConsoleMessage>(); 129 } 130 mErrorMessageCache.add(new ErrorConsoleMessage(msg, sourceId, lineNumber)); 131 } 132 } 133 134 /** 135 * Removes all error messages from the console. 136 */ clearErrorMessages()137 public void clearErrorMessages() { 138 if (mSetupComplete) { 139 mErrorList.clearErrorMessages(); 140 } else if (mErrorMessageCache != null) { 141 mErrorMessageCache.clear(); 142 } 143 } 144 145 /** 146 * Returns the current number of errors displayed in the console. 147 */ numberOfErrors()148 public int numberOfErrors() { 149 if (mSetupComplete) { 150 return mErrorList.getCount(); 151 } else { 152 return (mErrorMessageCache == null) ? 0 : mErrorMessageCache.size(); 153 } 154 } 155 156 /** 157 * Sets the webview that this console is associated with. Currently this is used so 158 * we can call into webkit to evaluate JS expressions in the console. 159 */ setWebView(WebView webview)160 public void setWebView(WebView webview) { 161 mWebView = webview; 162 } 163 164 /** 165 * Sets the visibility state of the console. 166 */ showConsole(int show_state)167 public void showConsole(int show_state) { 168 commonSetupIfNeeded(); 169 switch (show_state) { 170 case SHOW_MINIMIZED: 171 mConsoleHeader.setVisibility(View.VISIBLE); 172 mConsoleHeader.setText(R.string.error_console_header_text_minimized); 173 mErrorList.setVisibility(View.GONE); 174 mEvalJsViewGroup.setVisibility(View.GONE); 175 break; 176 177 case SHOW_MAXIMIZED: 178 mConsoleHeader.setVisibility(View.VISIBLE); 179 mConsoleHeader.setText(R.string.error_console_header_text_maximized); 180 mErrorList.setVisibility(View.VISIBLE); 181 mEvalJsViewGroup.setVisibility(View.VISIBLE); 182 break; 183 184 case SHOW_NONE: 185 mConsoleHeader.setVisibility(View.GONE); 186 mErrorList.setVisibility(View.GONE); 187 mEvalJsViewGroup.setVisibility(View.GONE); 188 break; 189 } 190 mCurrentShowState = show_state; 191 } 192 193 /** 194 * Returns the current visibility state of the console. 195 */ getShowState()196 public int getShowState() { 197 if (mSetupComplete) { 198 return mCurrentShowState; 199 } else { 200 return SHOW_NONE; 201 } 202 } 203 204 /** 205 * This class extends ListView to implement the View that will actually display the set of 206 * errors encountered on the current page. 207 */ 208 private static class ErrorConsoleListView extends ListView { 209 // An adapter for this View that contains a list of error messages. 210 private ErrorConsoleMessageList mConsoleMessages; 211 ErrorConsoleListView(Context context, AttributeSet attributes)212 public ErrorConsoleListView(Context context, AttributeSet attributes) { 213 super(context, attributes); 214 mConsoleMessages = new ErrorConsoleMessageList(context); 215 setAdapter(mConsoleMessages); 216 } 217 addErrorMessage(String msg, String sourceId, int lineNumber)218 public void addErrorMessage(String msg, String sourceId, int lineNumber) { 219 mConsoleMessages.add(msg, sourceId, lineNumber); 220 setSelection(mConsoleMessages.getCount()); 221 } 222 clearErrorMessages()223 public void clearErrorMessages() { 224 mConsoleMessages.clear(); 225 } 226 227 /** 228 * This class is an adapter for ErrorConsoleListView that contains the error console 229 * message data. 230 */ 231 private class ErrorConsoleMessageList extends android.widget.BaseAdapter 232 implements android.widget.ListAdapter { 233 234 private Vector<ErrorConsoleMessage> mMessages; 235 private LayoutInflater mInflater; 236 ErrorConsoleMessageList(Context context)237 public ErrorConsoleMessageList(Context context) { 238 mMessages = new Vector<ErrorConsoleMessage>(); 239 mInflater = (LayoutInflater)context.getSystemService( 240 Context.LAYOUT_INFLATER_SERVICE); 241 } 242 243 /** 244 * Add a new message to the list and update the View. 245 */ add(String msg, String sourceID, int lineNumber)246 public void add(String msg, String sourceID, int lineNumber) { 247 mMessages.add(new ErrorConsoleMessage(msg, sourceID, lineNumber)); 248 notifyDataSetChanged(); 249 } 250 251 /** 252 * Remove all messages from the list and update the view. 253 */ clear()254 public void clear() { 255 mMessages.clear(); 256 notifyDataSetChanged(); 257 } 258 259 @Override areAllItemsEnabled()260 public boolean areAllItemsEnabled() { 261 return false; 262 } 263 264 @Override isEnabled(int position)265 public boolean isEnabled(int position) { 266 return false; 267 } 268 getItemId(int position)269 public long getItemId(int position) { 270 return position; 271 } 272 getItem(int position)273 public Object getItem(int position) { 274 return mMessages.get(position); 275 } 276 getCount()277 public int getCount() { 278 return mMessages.size(); 279 } 280 281 @Override hasStableIds()282 public boolean hasStableIds() { 283 return true; 284 } 285 286 /** 287 * Constructs a TwoLineListItem for the error at position. 288 */ getView(int position, View convertView, ViewGroup parent)289 public View getView(int position, View convertView, ViewGroup parent) { 290 View view; 291 ErrorConsoleMessage error = mMessages.get(position); 292 293 if (error == null) { 294 return null; 295 } 296 297 if (convertView == null) { 298 view = mInflater.inflate(android.R.layout.two_line_list_item, parent, false); 299 } else { 300 view = convertView; 301 } 302 303 TextView headline = (TextView) view.findViewById(android.R.id.text1); 304 TextView subText = (TextView) view.findViewById(android.R.id.text2); 305 headline.setText(error.getSourceID() + ":" + error.getLineNumber()); 306 subText.setText(error.getMessage()); 307 return view; 308 } 309 310 } 311 } 312 313 /** 314 * This class holds the data for a single error message in the console. 315 */ 316 private static class ErrorConsoleMessage { 317 private String mMessage; 318 private String mSourceID; 319 private int mLineNumber; 320 ErrorConsoleMessage(String msg, String sourceID, int lineNumber)321 public ErrorConsoleMessage(String msg, String sourceID, int lineNumber) { 322 mMessage = msg; 323 mSourceID = sourceID; 324 mLineNumber = lineNumber; 325 } 326 getMessage()327 public String getMessage() { 328 return mMessage; 329 } 330 getSourceID()331 public String getSourceID() { 332 return mSourceID; 333 } 334 getLineNumber()335 public int getLineNumber() { 336 return mLineNumber; 337 } 338 } 339 } 340