blob: 356bcfc208d7bc82fef8a7f7f3c9b05c6ea995a2 [file] [log] [blame]
/*
* Copyright (C) 2007 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.stk;
import android.app.ActionBar;
import android.app.Activity;
import android.app.AlarmManager;
import android.content.Context;
import android.content.Intent;
import android.content.res.Configuration;
import android.os.Bundle;
import android.os.SystemClock;
import android.text.Editable;
import android.text.InputFilter;
import android.text.TextUtils;
import android.text.TextWatcher;
import android.text.method.PasswordTransformationMethod;
import android.view.KeyEvent;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.WindowManager;
import android.view.inputmethod.EditorInfo;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.PopupMenu;
import android.widget.TextView;
import android.widget.TextView.BufferType;
import com.android.internal.telephony.cat.CatLog;
import com.android.internal.telephony.cat.Input;
/**
* Display a request for a text input a long with a text edit form.
*/
public class StkInputActivity extends Activity implements View.OnClickListener,
TextWatcher {
// Members
private int mState;
private EditText mTextIn = null;
private TextView mPromptView = null;
private View mMoreOptions = null;
private PopupMenu mPopupMenu = null;
private View mYesNoLayout = null;
private View mNormalLayout = null;
// Constants
private static final String className = new Object(){}.getClass().getEnclosingClass().getName();
private static final String LOG_TAG = className.substring(className.lastIndexOf('.') + 1);
private Input mStkInput = null;
// Constants
private static final int STATE_TEXT = 1;
private static final int STATE_YES_NO = 2;
static final String YES_STR_RESPONSE = "YES";
static final String NO_STR_RESPONSE = "NO";
// Font size factor values.
static final float NORMAL_FONT_FACTOR = 1;
static final float LARGE_FONT_FACTOR = 2;
static final float SMALL_FONT_FACTOR = (1 / 2);
// Keys for saving the state of the activity in the bundle
private static final String RESPONSE_SENT_KEY = "response_sent";
private static final String INPUT_STRING_KEY = "input_string";
private static final String ALARM_TIME_KEY = "alarm_time";
private static final String PENDING = "pending";
private static final String INPUT_ALARM_TAG = LOG_TAG;
private static final long NO_INPUT_ALARM = -1;
private long mAlarmTime = NO_INPUT_ALARM;
private StkAppService appService = StkAppService.getInstance();
private boolean mIsResponseSent = false;
// Determines whether this is in the pending state.
private boolean mIsPending = false;
private int mSlotId = -1;
// Click listener to handle buttons press..
public void onClick(View v) {
String input = null;
if (mIsResponseSent) {
CatLog.d(LOG_TAG, "Already responded");
return;
}
switch (v.getId()) {
case R.id.button_ok:
input = mTextIn.getText().toString();
break;
case R.id.button_cancel:
sendResponse(StkAppService.RES_ID_END_SESSION);
finish();
return;
// Yes/No layout buttons.
case R.id.button_yes:
input = YES_STR_RESPONSE;
break;
case R.id.button_no:
input = NO_STR_RESPONSE;
break;
case R.id.more:
if (mPopupMenu == null) {
mPopupMenu = new PopupMenu(this, v);
Menu menu = mPopupMenu.getMenu();
createOptionsMenuInternal(menu);
prepareOptionsMenuInternal(menu);
mPopupMenu.setOnMenuItemClickListener(new PopupMenu.OnMenuItemClickListener() {
public boolean onMenuItemClick(MenuItem item) {
optionsItemSelectedInternal(item);
return true;
}
});
mPopupMenu.setOnDismissListener(new PopupMenu.OnDismissListener() {
public void onDismiss(PopupMenu menu) {
mPopupMenu = null;
}
});
mPopupMenu.show();
}
return;
default:
break;
}
CatLog.d(LOG_TAG, "handleClick, ready to response");
sendResponse(StkAppService.RES_ID_INPUT, input, false);
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
CatLog.d(LOG_TAG, "onCreate - mIsResponseSent[" + mIsResponseSent + "]");
// appService can be null if this activity is automatically recreated by the system
// with the saved instance state right after the phone process is killed.
if (appService == null) {
CatLog.d(LOG_TAG, "onCreate - appService is null");
finish();
return;
}
ActionBar actionBar = null;
if (getResources().getBoolean(R.bool.show_menu_title_only_on_menu)) {
actionBar = getActionBar();
if (actionBar != null) {
actionBar.hide();
}
}
// Set the layout for this activity.
setContentView(R.layout.stk_input);
if (actionBar != null) {
mMoreOptions = findViewById(R.id.more);
mMoreOptions.setVisibility(View.VISIBLE);
mMoreOptions.setOnClickListener(this);
}
// Initialize members
mTextIn = (EditText) this.findViewById(R.id.in_text);
mPromptView = (TextView) this.findViewById(R.id.prompt);
// Set buttons listeners.
Button okButton = (Button) findViewById(R.id.button_ok);
Button cancelButton = (Button) findViewById(R.id.button_cancel);
Button yesButton = (Button) findViewById(R.id.button_yes);
Button noButton = (Button) findViewById(R.id.button_no);
okButton.setOnClickListener(this);
cancelButton.setOnClickListener(this);
yesButton.setOnClickListener(this);
noButton.setOnClickListener(this);
mYesNoLayout = findViewById(R.id.yes_no_layout);
mNormalLayout = findViewById(R.id.normal_layout);
initFromIntent(getIntent());
}
@Override
protected void onPostCreate(Bundle savedInstanceState) {
super.onPostCreate(savedInstanceState);
mTextIn.addTextChangedListener(this);
}
@Override
public void onResume() {
super.onResume();
CatLog.d(LOG_TAG, "onResume - mIsResponseSent[" + mIsResponseSent +
"], slot id: " + mSlotId);
// If the terminal has already sent response to the card when this activity is resumed,
// keep this as a pending activity as this should be finished when the session ends.
if (!mIsResponseSent) {
setPendingState(false);
}
if (mAlarmTime == NO_INPUT_ALARM) {
startTimeOut();
}
}
@Override
public void onPause() {
super.onPause();
CatLog.d(LOG_TAG, "onPause - mIsResponseSent[" + mIsResponseSent + "]");
if (mPopupMenu != null) {
mPopupMenu.dismiss();
}
}
@Override
public void onStop() {
super.onStop();
CatLog.d(LOG_TAG, "onStop - mIsResponseSent[" + mIsResponseSent + "]");
// Nothing should be done here if this activity is being finished or restarted now.
if (isFinishing() || isChangingConfigurations()) {
return;
}
if (mIsResponseSent) {
// It is unnecessary to keep this activity if the response was already sent and
// the dialog activity is NOT on the top of this activity.
if (!appService.isStkDialogActivated()) {
finish();
}
} else {
// This should be registered as the pending activity here
// only when no response has been sent back to the card.
setPendingState(true);
}
}
@Override
public void onDestroy() {
super.onDestroy();
CatLog.d(LOG_TAG, "onDestroy - before Send End Session mIsResponseSent[" +
mIsResponseSent + " , " + mSlotId + "]");
if (appService == null) {
return;
}
// Avoid sending the terminal response while the activty is being restarted
// due to some kind of configuration change.
if (!isChangingConfigurations()) {
// If the input activity is finished by stkappservice
// when receiving OP_LAUNCH_APP from the other SIM, we can not send TR here,
// since the input cmd is waiting user to process.
if (!mIsResponseSent && !appService.isInputPending(mSlotId)) {
CatLog.d(LOG_TAG, "handleDestroy - Send End Session");
sendResponse(StkAppService.RES_ID_END_SESSION);
}
}
cancelTimeOut();
}
@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
if (mPopupMenu != null) {
mPopupMenu.dismiss();
}
}
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
if (mIsResponseSent) {
CatLog.d(LOG_TAG, "Already responded");
return true;
}
switch (keyCode) {
case KeyEvent.KEYCODE_BACK:
CatLog.d(LOG_TAG, "onKeyDown - KEYCODE_BACK");
sendResponse(StkAppService.RES_ID_BACKWARD, null, false);
return true;
}
return super.onKeyDown(keyCode, event);
}
void sendResponse(int resId) {
sendResponse(resId, null, false);
}
void sendResponse(int resId, String input, boolean help) {
cancelTimeOut();
if (mSlotId == -1) {
CatLog.d(LOG_TAG, "slot id is invalid");
return;
}
if (StkAppService.getInstance() == null) {
CatLog.d(LOG_TAG, "StkAppService is null, Ignore response: id is " + resId);
return;
}
if (mMoreOptions != null) {
mMoreOptions.setVisibility(View.INVISIBLE);
}
CatLog.d(LOG_TAG, "sendResponse resID[" + resId + "] input[*****] help["
+ help + "]");
mIsResponseSent = true;
Bundle args = new Bundle();
args.putInt(StkAppService.RES_ID, resId);
if (input != null) {
args.putString(StkAppService.INPUT, input);
}
args.putBoolean(StkAppService.HELP, help);
appService.sendResponse(args, mSlotId);
// This instance should be set as a pending activity and finished by the service
if (resId != StkAppService.RES_ID_END_SESSION) {
setPendingState(true);
}
}
@Override
public boolean onCreateOptionsMenu(android.view.Menu menu) {
super.onCreateOptionsMenu(menu);
createOptionsMenuInternal(menu);
return true;
}
private void createOptionsMenuInternal(Menu menu) {
menu.add(Menu.NONE, StkApp.MENU_ID_END_SESSION, 1, R.string.menu_end_session);
menu.add(0, StkApp.MENU_ID_HELP, 2, R.string.help);
}
@Override
public boolean onPrepareOptionsMenu(android.view.Menu menu) {
super.onPrepareOptionsMenu(menu);
prepareOptionsMenuInternal(menu);
return true;
}
private void prepareOptionsMenuInternal(Menu menu) {
menu.findItem(StkApp.MENU_ID_END_SESSION).setVisible(true);
menu.findItem(StkApp.MENU_ID_HELP).setVisible(mStkInput.helpAvailable);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
if (optionsItemSelectedInternal(item)) {
return true;
}
return super.onOptionsItemSelected(item);
}
private boolean optionsItemSelectedInternal(MenuItem item) {
if (mIsResponseSent) {
CatLog.d(LOG_TAG, "Already responded");
return true;
}
switch (item.getItemId()) {
case StkApp.MENU_ID_END_SESSION:
sendResponse(StkAppService.RES_ID_END_SESSION);
finish();
return true;
case StkApp.MENU_ID_HELP:
sendResponse(StkAppService.RES_ID_INPUT, "", true);
return true;
}
return false;
}
@Override
protected void onSaveInstanceState(Bundle outState) {
CatLog.d(LOG_TAG, "onSaveInstanceState: " + mSlotId);
outState.putBoolean(RESPONSE_SENT_KEY, mIsResponseSent);
outState.putString(INPUT_STRING_KEY, mTextIn.getText().toString());
outState.putLong(ALARM_TIME_KEY, mAlarmTime);
outState.putBoolean(PENDING, mIsPending);
}
@Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
CatLog.d(LOG_TAG, "onRestoreInstanceState: " + mSlotId);
mIsResponseSent = savedInstanceState.getBoolean(RESPONSE_SENT_KEY);
if (mIsResponseSent && (mMoreOptions != null)) {
mMoreOptions.setVisibility(View.INVISIBLE);
}
String savedString = savedInstanceState.getString(INPUT_STRING_KEY);
mTextIn.setText(savedString);
updateButton();
mAlarmTime = savedInstanceState.getLong(ALARM_TIME_KEY, NO_INPUT_ALARM);
if (mAlarmTime != NO_INPUT_ALARM) {
startTimeOut();
}
if (!mIsResponseSent && !savedInstanceState.getBoolean(PENDING)) {
// If this is in the foreground and no response has been sent to the card,
// this must not be registered as pending activity by the previous instance.
// No need to renew nor clear pending activity in this case.
} else {
// Renew the instance of the pending activity.
setPendingState(true);
}
}
private void setPendingState(boolean on) {
if (mIsPending != on) {
appService.getStkContext(mSlotId).setPendingActivityInstance(on ? this : null);
mIsPending = on;
}
}
public void beforeTextChanged(CharSequence s, int start, int count,
int after) {
}
public void onTextChanged(CharSequence s, int start, int before, int count) {
// Reset timeout.
cancelTimeOut();
startTimeOut();
updateButton();
}
public void afterTextChanged(Editable s) {
}
private void updateButton() {
// Disable the button if the length of the input text does not meet the expectation.
Button okButton = (Button) findViewById(R.id.button_ok);
okButton.setEnabled((mTextIn.getText().length() < mStkInput.minLen) ? false : true);
}
private void cancelTimeOut() {
if (mAlarmTime != NO_INPUT_ALARM) {
CatLog.d(LOG_TAG, "cancelTimeOut - slot id: " + mSlotId);
AlarmManager am = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
am.cancel(mAlarmListener);
mAlarmTime = NO_INPUT_ALARM;
}
}
private void startTimeOut() {
// No need to set alarm if device sent TERMINAL RESPONSE already.
if (mIsResponseSent) {
return;
}
if (mAlarmTime == NO_INPUT_ALARM) {
int duration = StkApp.calculateDurationInMilis(mStkInput.duration);
if (duration <= 0) {
duration = StkApp.UI_TIMEOUT;
}
mAlarmTime = SystemClock.elapsedRealtime() + duration;
}
CatLog.d(LOG_TAG, "startTimeOut: " + mAlarmTime + "ms, slot id: " + mSlotId);
AlarmManager am = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
am.setExact(AlarmManager.ELAPSED_REALTIME_WAKEUP, mAlarmTime, INPUT_ALARM_TAG,
mAlarmListener, null);
}
private void configInputDisplay() {
TextView numOfCharsView = (TextView) findViewById(R.id.num_of_chars);
TextView inTypeView = (TextView) findViewById(R.id.input_type);
int inTypeId = R.string.alphabet;
// set the prompt.
if ((mStkInput.icon == null || !mStkInput.iconSelfExplanatory)
&& !TextUtils.isEmpty(mStkInput.text)) {
mPromptView.setText(mStkInput.text);
mPromptView.setVisibility(View.VISIBLE);
}
// Set input type (alphabet/digit) info close to the InText form.
if (mStkInput.digitOnly) {
mTextIn.setKeyListener(StkDigitsKeyListener.getInstance());
inTypeId = R.string.digits;
}
inTypeView.setText(inTypeId);
setTitle(R.string.app_name);
if (mStkInput.icon != null) {
ImageView imageView = (ImageView) findViewById(R.id.icon);
imageView.setImageBitmap(mStkInput.icon);
imageView.setVisibility(View.VISIBLE);
}
// Handle specific global and text attributes.
switch (mState) {
case STATE_TEXT:
int maxLen = mStkInput.maxLen;
int minLen = mStkInput.minLen;
mTextIn.setFilters(new InputFilter[] {new InputFilter.LengthFilter(
maxLen)});
// Set number of chars info.
String lengthLimit = String.valueOf(minLen);
if (maxLen != minLen) {
lengthLimit = minLen + " - " + maxLen;
}
numOfCharsView.setText(lengthLimit);
if (!mStkInput.echo) {
mTextIn.setTransformationMethod(PasswordTransformationMethod
.getInstance());
}
mTextIn.setImeOptions(EditorInfo.IME_FLAG_NO_FULLSCREEN);
// Request the initial focus on the edit box and show the software keyboard.
mTextIn.requestFocus();
getWindow().setSoftInputMode(
WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE);
// Set default text if present.
if (mStkInput.defaultText != null) {
mTextIn.setText(mStkInput.defaultText);
} else {
// make sure the text is cleared
mTextIn.setText("", BufferType.EDITABLE);
}
updateButton();
break;
case STATE_YES_NO:
// Set display mode - normal / yes-no layout
mYesNoLayout.setVisibility(View.VISIBLE);
mNormalLayout.setVisibility(View.GONE);
break;
}
}
private void initFromIntent(Intent intent) {
// Get the calling intent type: text/key, and setup the
// display parameters.
CatLog.d(LOG_TAG, "initFromIntent - slot id: " + mSlotId);
if (intent != null) {
mStkInput = intent.getParcelableExtra("INPUT");
mSlotId = intent.getIntExtra(StkAppService.SLOT_ID, -1);
CatLog.d(LOG_TAG, "onCreate - slot id: " + mSlotId);
if (mStkInput == null) {
finish();
} else {
mState = mStkInput.yesNo ? STATE_YES_NO :
STATE_TEXT;
configInputDisplay();
}
} else {
finish();
}
}
private final AlarmManager.OnAlarmListener mAlarmListener =
new AlarmManager.OnAlarmListener() {
@Override
public void onAlarm() {
CatLog.d(LOG_TAG, "The alarm time is reached");
mAlarmTime = NO_INPUT_ALARM;
sendResponse(StkAppService.RES_ID_TIMEOUT);
}
};
}