继续上一章,我们看了一下基本的目录结构。那么现在我们要把那个简单的Helloworld界面改的复杂一点了。
现在把helloworl改成一个登录界面。需要增加两个输入框和一个按钮:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity"
android:orientation="vertical">
<TextView android:text="@string/title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
/>
<EditText
android:id="@+id/username"
style="@style/EditTextStyle"
android:layout_height="wrap_content"
android:layout_width="match_parent"
android:hint="@string/username_hint"
android:imeActionId="@+id/username"
android:imeOptions="actionUnspecified"
android:inputType="text"/>
<EditText
android:id="@+id/password"
style="@style/EditTextStyle"
android:layout_height="wrap_content"
android:layout_width="match_parent"
android:hint="@string/password_hint"
android:imeActionId="@+id/password"
android:imeOptions="actionUnspecified"
android:inputType="textPassword"/>
<Button
android:id="@+id/button_login"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/button_login"
/>
</LinearLayout>
代码如上。我们首选要把 RelativeLayout
改成一个 LinearLayout
,并且把布局改成垂直的,这样就可以一行一个来布局了。然后增加两个输入框和一个按钮。
之前说过 @xxx
的语法是引用 res 中的资源,不过这里有一个特殊的地方 @+id/username
,其实在res中并没有定义一个username的字段。这里的加号表示不是引用 而是直接创建一个id。另外的一些属性看一下名字就知道什么意思了,不然就去官网上查。相应的, strings.xml 文件也要增加一些字段,这里就不贴出来了。
改完之后运行,会发现现在已经有一个很粗糙的登录界面了,只不过点击了没有任何反应。
现在我们希望在点击登录按钮之后能有一些反应,比如弹出一个对话框什么的。需要做的就是给Button绑定一个click事件。
protected void initialize() {
Button loginButton = (Button) findViewById(R.id.button_login);
final Editable username = ((EditText)findViewById(R.id.username)).getText();
loginButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(MainActivity.this, "您的用户名是:" + username, Toast.LENGTH_LONG).show();
}
});
}
参考上面的代码,几个重点:
findViewById(R.id.button_login)
可以找到你需要的控件,其中的id就是通过 @+id/xxx
定义的,不过这个函数总是返回一个View,所以你需要自己转换一下loginButton.setOnClickListener
来绑定一个点击事件,并且在其中调用toast弹出一个消息。到目前为止我们都在一个Activity中折腾,下面我们创建一个新的Activity:
创建一个Activity需要创建两个文件,一个是Java类,一个是对应的xml文件。不过Android Studio可以自动帮我们做这件事,选中java包,右键选择 new -> activity -> blank activity
然后输入一个名字即可。点击完成之后会自动两个文件,一个 ProfileActivy.java
和一个 activity_profile.xml
,并且会自动在 AndroidManifest.xml 中注册这个活动,如果你没有用IDE或者是想尝试下手动创建,那么你需要自己做这三件事。
然后我们希望当用户点击确定按钮的时候,我们打开这个新的activity并且把用户输入的内容传过去。
首先我们需要改一下 MainActivity.java 文件:
protected void initialize() {
Button loginButton = (Button) findViewById(R.id.button_login);
loginButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(MainActivity.this, ProfileActivity.class);
final String username = ((EditText)findViewById(R.id.username)).getText().toString();
intent.putExtra(PROFILE_MESSAGE, username);
startActivity(intent);
}
});
}
我们这里用到一个新的类叫 Intent
,这个Intent就是用来打开另外一个 Activity 用的(当然还有别的作用,不过主要是这个)。 它的两个参数分别是当前的 Context 和 需要打开的 Activity。
intent 有一个 putExtra
方法,可以以键值对的方式向下一个 activity 传递数据,这里我们就把用户输入的用户名传过去。
再打开 ProfileActivity.java 文件,我们需要接收intent传过来的参数,这样写:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_profile);
Intent intent = getIntent();
String message = intent.getStringExtra(MainActivity.PROFILE_MESSAGE);
TextView profileText = (TextView) findViewById(R.id.profile_text);
profileText.setText(message);
}
通过 getIntent()
方法我们就可以取到打开当前activity的intent,然后用 getStringExtra 来取传过来的参数。最后设置到 textview 中去即可。
上面 new Intent(MainActivity.this, ProfileActivity.class)
是通过显示的指定了class来创建一个intent。实际上这样会导致代码耦合,因为指定的class是一个实现类。我们更希望只要声明自己只需要指定一个名字,而不用关心是哪一个具体的类实现的,比如我们只要指定是 com.lihongxun.Profile
就行了,不想关心到底是哪个类实现的,而对应的activity也声明自己提供的是 com.lihongxun.Profile
就行了。
那么我们需要做的就是先在 AndroidManifest.xml 中的声明中指定一个名字:
<activity
android:name=".ProfileActivity"
android:label="@string/title_activity_profile" >
<intent-filter>
<action android:name="com.lihongxun.Profile" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
然后在java中通过名字调用即可:
Intent intent = new Intent("com.lihongxun.Profile");
可能看到这里还是觉得这种隐式调用很鸡肋,明明有class,没必要在申明一个名字。
其实这样做的最重要一点在于,你可以调用其他的应用的功能。只要知道一个名字就行了,因为不可能拿到别人的实现类。比如你可以调用支付宝的支付接口。
上面说到了通过 intent 可以向打开的下一个活动传递数据,那么同样的,也可以在当前活动关闭的时候向上一个活动返回数据。
向上一个活动返回数据,只需要做三处修改
一,在 MainActivity
中把 startActivity(intent);
改成 startActivityForResult(intent);
也就是声明要开始一个有返回数据的活动
二,在 MainActivity
中注册一个事件,用来接受返回的数据:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
switch (requestCode) {
case 1:
if (resultCode == RESULT_OK) {
String result = data.getStringExtra("result");
Toast.makeText(this, result, Toast.LENGTH_LONG).show();
}
break;
default:
}
}
三,在 ProfileActivity
中设置返回值,通过一个intent来设置返回值,代码如下:
@Override
public void onBackPressed() { //当点击回退按钮的时候触发
//向上一个活动返回数据
Intent intent = new Intent();
intent.putExtra("result", "I am the result");
setResult(RESULT_OK, intent);
finish();
}