Activity的跳转与传值:startActivityForResult,setResult

Activity跳转与传值,主要是通过Intent类来连接多个Activity,以及传递数据。

Intent是Android一个很重要的类。Intent直译是“意图”,什么是意图呢?比如你想从这个Activity跳转到另外一个Activity,这就是一个意图。

Activity跳转,无返回结果

这是最简单的Activity跳转方式。从一个Activity启动另一个Activity,直接

startActivity(new Intent(当前Activity.this, 下一Activity.class))。

((Button) findViewById(R.id.Notepadv1)).setOnClickListener( new OnClickListener() {     
    public void onClick(View v) {     
        startActivity(new Intent (MyAndroidAppActivity.this, Notepadv1.class) );     
    }     
});

Activity跳转,返回数据/结果

需要返回数据或结果的,则使用startActivityForResult (Intent intent, int requestCode),requestCode的值是自定义的,用于识别跳转的目标Activity。

跳转的目标Activity所要做的就是返回数据/结果,setResult(int resultCode)只返回结果不带数据,或者setResult(int resultCode, Intent data)两者都返回!

而接收返回的数据/结果的处理函数是onActivityResult(int requestCode, int resultCode, Intent data),这里的requestCode就是startActivityForResult的requestCode,resultCode就是setResult里面的resultCode,返回的数据在data里面。

MyAndroidAppActivity:

static final int SEND_SMS_REQUEST = 0;    
static final int CALL_REQUEST = 1;    
((Button) findViewById(R.id.sms)).setOnClickListener( new OnClickListener() {    
    public void onClick(View v) {    
        Intent intent = new Intent(MyAndroidAppActivity.this, SendSMSActivity.class);    
        startActivityForResult(intent, SEND_SMS_REQUEST);    
    }    
});    
@Override    
protected void onActivityResult(int requestCode, int resultCode, Intent data) {    
    if (requestCode == SEND_SMS_REQUEST) {    
        if (resultCode == RESULT_OK) {    
            Toast.makeText(this, "Send SMS RESULT_OK", Toast.LENGTH_SHORT).show();    
        }else if (resultCode == RESULT_CANCELED) {    
            Bundle bundle = data.getExtras();    
            String phoneno = bundle.getString("phoneNO");     
            Toast.makeText(this, "Send SMS RESULT_CANCELED "+phoneno, Toast.LENGTH_SHORT).show();    
        }    
    }else if (requestCode == CALL_REQUEST) {    
        if (resultCode == RESULT_CANCELED) {    
            Toast.makeText(this, "Call RESULT_CANCELED", Toast.LENGTH_SHORT).show();    
        }    
    }    
}

SendSMSActivity:

((Button) findViewById(R.id.send)).setOnClickListener( new Button.OnClickListener() {    
    public void onClick(View v) {    
        SendSMSActivity.this.setResult(RESULT_OK);    
        SendSMSActivity.this.finish();    
    }    
}):

**** 注意,在setResult后,要调用finish()销毁当前的Activity,否则无法返回到原来的Activity,就无法执行原来Activity的onActivityResult函数,看到当前的Activity没反应。**

RESULT_OK和RESULT_CANCELED是系统空间里面的常量,不需要自定义,直接使用就行。

另外在运行过程中,发现按Back键后,是可以返回RESULT_CANCELED的,看了SDK doc后,原来真的是那样,而且不带有数据的。这意味着,如果你设想在返回RESULT_CANCELED时并返回数据,那么需要截获Back键的事件处理,把原来返回RESULT_CANCELED的核心逻辑copy到事件处理里面。这里给个例子:

((Button) findViewById(R.id.cancel)).setOnClickListener( new Button.OnClickListener() {    
    public void onClick(View v) {    
        // 实例化 Bundle,设置需要传递的参数    
        Bundle bundle = new Bundle();    
        bundle.putString("phoneNO", "020-123");    
        SendSMSActivity.this.setResult(RESULT_CANCELED, SendSMSActivity.this.getIntent().putExtras(bundle));    
        SendSMSActivity.this.finish();    
    }    
});    
@Override    
public boolean onKeyDown(int keyCode, KeyEvent event) {    
    // 是否触发按键为back键       
    if (keyCode == KeyEvent.KEYCODE_BACK) {    
        // 实例化 Bundle,设置需要传递的参数    
        Bundle bundle = new Bundle();    
        bundle.putString("phoneNO", "020-123");    
        setResult(RESULT_CANCELED, this.getIntent().putExtras(bundle));    
        this.finish();    
        return true;    
    }else {    
        return super.onKeyDown(keyCode, event);    
    }    
}

这里要注意的是,在处理Back键事件后return true则表示本事件不再传递给其他函数处理,可理解为由当前函数全权负责处理,所以在return前finish当前Activity,保持原来的行为表现,当然我们可以设置为隐藏当前Activity等其他行为。

Activity传送数据

在上面的代码中,我们可以看到使用Bundle来存储数据,并将其putExtras到Intent里面。Bundle使用“名字-值”来存储数据。那么,从原来的Activity A传送数据到新的Activity B, Activity B如何获取传送过来的Intent参数时呢?最近写了这样的一个例子:

In Activity A:

// 在某个按钮响应事件里    
Intent intent = new Intent(this, TextInputActivity.class);    
intent.putExtra("Text", mText);    
intent.putExtra("TextColor", mTextColor);    
intent.putExtra("TextSize", mTextSize);    
intent.putExtra("TextBold", mTextBold);    
startActivityForResult(intent, REQUEST_TEXT);

In Activity B:

// in onCreate(Bundle savedInstanceState)    
Bundle extras = getIntent().getExtras();    
mText = extras.getString("Text");    
mTextColor = extras.getInt("TextColor");    
mTextSize = extras.getFloat("TextSize");    
mTextBold = extras.getBoolean("TextBold");