这节课是 Android 开发(入门)课程 的第一部分《布局和交互》的最后一节课,导师是 Katherine Kuan 和 Kunal Chawla,主要内容是更多的 XML 布局与 Java 代码,完善 Just Java App。这节课出现了很多第一次应用的 Views,但在掌握了高效的学习方法的情况下,可以体验到快速上手的感觉,下面就带领大家开始。
关键词:CheckBox、ScrollView、EditText、strings.xml、styles.xml,if-else 流控语句、Intent
CheckBox
Google 搜索 "checkbox android" 第一个结果是 Android 文档;第二个结果是代码示例,也可以在 Common Android Views Cheat Sheets 中找到。
Just Java App 中的 CheckBox 代码如下:
<CheckBox
android:id="@+id/whipped_cream_checkbox"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginBottom="8dp"
android:paddingEnd="24dp"
android:paddingStart="24dp"
android:text="@string/whipped_cream"
android:textSize="16sp" />
CheckBox 是 TextView 的子 class,所以拥有 TextView 的属性,如 android:text
。其中,
android:paddingEnd="24dp"
android:paddingStart="24dp"
将勾选框和文字分开 24dp。paddingStart 应用于从左往右的视图,paddingEnd 应用于从右往左的视图。
CheckBox 是一个对象,它有一个布尔 (Boolean) 类型的的变量来保存勾选框的状态,显然只有真 (true) 或假 (false) 两种,也就是说布尔变量只有两个值。Google 搜索 "java data type" 查找 Java 的基本数据类型 (Primitive Data Types) 进一步了解。
Just Java App 中的 CheckBox 对象如下:
CheckBox whippedCreamCheckBox = (CheckBox) findViewById(R.id.whipped_cream_checkbox);
boolean hasWhippedCream = whippedCreamCheckBox.isChecked();
//Log.v("MainActivity", "Has whipped cream:" + hasWhippedCream);
(1)布尔变量名同样遵循变量命名规则,常以 has 或 is 开头,如 hasWhippedCream。
(2)在调试时将信息打印到 logcat 中,能检查代码的运行情况。格式:
Log.v(“class名”, “字符串”);
- 注意打印信息的长度,避免刷屏刷掉错误等重要信息。调试结束后可将此行注释掉。
- 布尔变量与字符串连接会变成字符串 true 或 false,如 "Has whipped cream: " + hasWhippedCream → Has whipped cream: true 或 Has whipped cream: false。
ScrollView
Google 搜索 "how to scroll in android app" 找到 stack overflow 论坛的结果:利用 ScrollView 实现屏幕垂直方向滚动。这对设备横屏状态下非常有用。ScrollView 只能有一个子 View,Just Java App 中的 ScrollView 如下:
<ScrollView 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"
tools:context="com.example.android.justjava.MainActivity">
…
</ScrollView>
fill_parent 是 match_parent 的旧版本写法。
EditText
Google 搜索 "user input android" 找到 Best Practices for User Input 结果,选择 Keyboard Input 可找到 EditText 的代码示例,也可以在 Common Android Views Cheat Sheets 中找到。
EditText 允许用户在 App 中输入文字,Just Java App 中的 EditText 如下:
<EditText
android:id="@+id/name_field"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="@string/name"
android:inputType="textCapWords" />
在 Android 文档中查找 EditText 的属性含义。
-
android:hint
设置在 EditText 无输入时显示的提示文字。 -
android:inputType
设置文字在 EditText 中的显示方式,textCapWords 表示每个单词首字母自动大写,但对 Google 拼音输入法无效。
Google 搜索 "how do i get text from edittext field android" 找到 stack overflow 论坛的结果:通过链式调用 EditText 的 methods 来获取 EditText 中的文字。
Just Java App 中的 EditText 对象如下:
EditText nameField = (EditText) findViewById(R.id.name_field);
String name = nameField.getText().toString();
//Log.v("MainActivity", "Name: " + name);
getText() 的返回值类型为 Editable 对象,进一步调用 toString() 确定返回值为字符串。
strings.xml
- 在 Android Studio 中左侧 Project 标签 Android 视图下 app→res→values→strings.xml 文件应保存 App 的所有字符串资源,所以应将先前硬编码的字符串取到 strings.xml 中。
- Just Java App 中的 strings.xml 代码如下:
<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
<!-- Title for the application. [CHAR LIMIT=12] -->
<string name="app_name">Just Java</string>
<!-- Hint text display in the empty field for the user's name [CHAR LIMIT=20] -->
<string name="name">Name</string>
<string name="toppings">Toppings</string>
<string name="whipped_cream">Whipped Cream</string>
<string name="chocolate">Chocolate</string>
<string name="quantity">Quantity</string>
<string name="decrement">-</string>
<string name="initial_quantity_value">1</string>
<string name="increment">+</string>
<string name="order">Order</string>
<!--
Name for the order summary. It will be shown in the format of "Name: Amy" where Amy is the
user's name. [CHAR LIMIT=NONE]
-->
<string name="order_summary_name">Name: <xliff:g id="name" example="Amy">%s</xliff:g></string>
<!--
Whipped cream topping for the order summary. It will be shown in the format of
"Add whipped cream? true" or "Add whipped cream? false". [CHAR LIMIT=NONE]
-->
<string name="order_summary_whipped_cream">Add Whipped Cream? <xliff:g id="addWhippedCream" example="true">%b</xliff:g></string>
<!--
Chocolate topping for the order summary. It will be shown in the format of
"Add chocolate? true" or "Add chocolate? false". [CHAR LIMIT=NONE]
-->
<string name="order_summary_chocolate">Add chocolate? <xliff:g id="addChocolate" example="true">%b</xliff:g></string>
<!--
Quantity of coffee cups for the order summary. It will be shown in the format of
"Quantity: 2", where 2 is the number of cups ordered. [CHAR LIMIT=NONE]
-->
<string name="order_summary_quantity">Quantity: <xliff:g id="quantity" example="2">%d</xliff:g></string>
<!--
Total price for the order summary. It will be shown in the format of
"Total: $10" where $10 is the price. [CHAR LIMIT=NONE]
-->
<string name="order_summary_price">Total: <xliff:g id="price" example="$10">%s</xliff:g></string>
<!-- Thank you message for the order summary. [CHAR LIMIT=NONE] -->
<string name="thank_you">Thank you!</string>
<!--
Subject line for the order summary email. It will be in the format of
"Just Java order for Amy" where Amy is the user's name. [CHAR LIMIT=NONE]
-->
<string name="order_summary_email_subject">Just Java order for <xliff:g id="name" example="Amy">%s</xliff:g></string>
</resources>
(1)所有内容在 <resources …>... </resources>
内。
(2)字符串定义格式:
<string name=“字符串名”>字符串</string>
(3)这里使用了 xliff 语句在字符串中添加变量,同时作为占位符表示这部分内容无需翻译,如
<!-- 指定 xliff 命名空间的 URL -->
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"
<!-- 在字符串中添加“姓名”字符串变量 -->
<string name="order_summary_name">Name: <xliff:g id="name" example="Amy">%s</xliff:g></string>
<!-- 在字符串中添加布尔类型变量 -->
<string name="order_summary_chocolate">Add chocolate? <xliff:g id="addChocolate" example="true">%b</xliff:g></string>
从上面的例子可以看出,xliff 语句起止于 <xliff:g ...>...</xliff:g>
,中间内容依次为
- 变量 ID,如
id="name"
- 变量内容示例,如
example="Amy"
- 变量格式说明符 (Format Specifiers),如 %s 指字符串,%b 指布尔类型,这篇文章列出了更多 Java 格式说明符。
(4)XML注释格式:
<!-- … -->
写下具体的注释使代码易理解,对于字符串复用和翻译很有帮助。
- 在 strings.xml 定义所有字符串资源后,可在 XML 和 Java 中调用。
(1)XML 调用字符串,格式:@string/字符串名
<TextView
android:id="@+id/quantity_text_view"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:paddingLeft="8dp"
android:paddingRight="8dp"
android:text="@string/initial_quantity_value"
android:textColor="#000000"
android:textSize="16sp" />
(2)Java 调用字符串,格式:R.string.字符串名
/**
* Create summary of the order..
*
* @param name of the Customer
* @param price of the order
* @param addWhippedCream is whether or not the user wants whipped cream topping
* @param addChocolate is whether or not the user wants chocolate topping
* @return text summary
*/
private String createOrderSummary(String name, int price, boolean addWhippedCream, boolean addChocolate) {
return getString(R.string.order_summary_name, name) + "\n" +
getString(R.string.order_summary_whipped_cream, addWhippedCream) + "\n" +
getString(R.string.order_summary_chocolate, addChocolate) + "\n" +
getString(R.string.order_summary_quantity, quantity) + "\n" +
getString(R.string.order_summary_price, NumberFormat.getCurrencyInstance().format(price)) + "\n" +
getString(R.string.thank_you);
}
- Android App 有默认 (Default) 资源和替代 (Alternate) 资源,使 App 可面对不同的运行环境加载不同的资源,Google 搜索 "localization checklist" 进一步了解。支持多语言是本地化的其中一项工作,Google 搜索 "iso 639-1" 找到 Wikipedia 结果,了解全球语言缩写,如中文为 zh。
- 在 app→res 路径下新建 values-zh 目录并新建 strings.xml,翻译字符串为中文:
<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
<!-- Title for the application. [CHAR LIMIT=12] -->
<string name="app_name">Just Java</string>
<!-- Hint text display in the empty field for the user's name [CHAR LIMIT=20] -->
<string name="name">姓名</string>
<string name="toppings">配料</string>
<string name="whipped_cream">奶盖</string>
<string name="chocolate">巧克力</string>
<string name="quantity">数量</string>
<string name="decrement">-</string>
<string name="initial_quantity_value">1</string>
<string name="increment">+</string>
<string name="order">下单</string>
<!--
Name for the order summary. It will be shown in the format of "Name: Amy" where Amy is the
user's name. [CHAR LIMIT=NONE]
-->
<string name="order_summary_name">姓名: <xliff:g id="name" example="Amy">%s</xliff:g></string>
<!--
Whipped cream topping for the order summary. It will be shown in the format of
"Add whipped cream? true" or "Add whipped cream? false". [CHAR LIMIT=NONE]
-->
<string name="order_summary_whipped_cream">需要添加奶盖? <xliff:g id="addWhippedCream" example="true">%b</xliff:g></string>
<!--
Chocolate topping for the order summary. It will be shown in the format of
"Add chocolate? true" or "Add chocolate? false". [CHAR LIMIT=NONE]
-->
<string name="order_summary_chocolate">需要添加巧克力? <xliff:g id="addChocolate" example="true">%b</xliff:g></string>
<!--
Quantity of coffee cups for the order summary. It will be shown in the format of
"Quantity: 2", where 2 is the number of cups ordered. [CHAR LIMIT=NONE]
-->
<string name="order_summary_quantity">数量:<xliff:g id="quantity" example="2">%d</xliff:g></string>
<!--
Total price for the order summary. It will be shown in the format of
"Total: $10" where $10 is the price. [CHAR LIMIT=NONE]
-->
<string name="order_summary_price">价格:<xliff:g id="price" example="$10">%s</xliff:g></string>
<!-- Thank you message for the order summary. [CHAR LIMIT=NONE] -->
<string name="thank_you">谢谢!</string>
<!--
Subject line for the order summary email. It will be in the format of
"Just Java order for Amy" where Amy is the user's name. [CHAR LIMIT=NONE]
-->
<string name="order_summary_email_subject">Just Java 订单 <xliff:g id="name" example="Amy">%s</xliff:g></string>
</resources>
在 Android Studio 中左侧 Project 标签 Android 视图下找到 app→res→values→strings.xml 文件,右键菜单中找到并点击 "Open Editor",可打开对照翻译界面,如下图所示。
更多资源本地化的信息可到 Localizing with Resources | Android Developers 网站查看。
styles.xml
- 在 Android Studio 中左侧 Project 标签 Android 视图下 app→res→values→styles.xml 文件可统一定义 App 的主题以及 Views 的样式,减少代码冗余。
- Just Java App 中的 styles.xml 代码如下:
<resources>
<!-- Base application theme. -->
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<!-- Customize your theme here. -->
<item name="colorPrimary">@color/colorPrimary</item>
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
<item name="colorAccent">@color/colorAccent</item>
</style>
<!-- Style for a header TextView. -->
<style name="HeaderTextStyle" parent="Theme.AppCompat.Light.DarkActionBar">
<!-- Customize your theme here. -->
<item name="android:layout_width">wrap_content</item>
<item name="android:layout_height">48dp</item>
<item name="android:gravity">center_vertical</item>
<item name="android:textAllCaps">true</item>
<item name="android:textSize">15sp</item>
</style>
</resources>
(1)所有内容在 <resources …>... </resources>
内。
(2)同一样式在 <style …>... </style>
内。
(3)属性定义格式:<item name=“属性名”>真实值</item>
- 在 styles.xml 定义样式资源后,可在 XML 中调用:
<TextView
style="@style/HeaderTextStyle"
android:text="@string/toppings" />
<TextView
style="@style/HeaderTextStyle"
android:text="@string/quantity" />
两个 TextView 相同处应用同一样式,不同处自行定义属性,减少代码冗余。
- 样式可应用于单个的 View,而主题的 style 是应用于某类活动或整个应用的,例如所有文字的颜色,应用栏或状态栏的颜色。如
<!-- Base application theme. -->
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<!-- Customize your theme here. -->
<item name="colorPrimary">@color/colorPrimary</item>
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
<item name="colorAccent">@color/colorAccent</item>
</style>
其中 parent="Theme.AppCompat.Light.DarkActionBar"
表示 App 主题通过 Appcompat 兼容使用了 Light.DarkActionBar 的主题。同时 AppTheme 在 AndroidManifest.xml 内引用,对于多屏幕应用,不同主题的 Activity 就在这个文件内指定。
if-else 语句
- 控制代码执行顺序称为控制流 (Control Flow),利用 if-else 语句可以使程序跳过部分代码。
- Just Java App 中的 if-else 语句如下:
/**
* This method is called when the + button is clicked.
*/
public void increment(View view) {
if (quantity == 100) {
// Show an error message as a toast
Toast.makeText(this,
"You cannot have more than 100 coffees!", Toast.LENGTH_SHORT).show();
// Exit this method early because there's nothing left to do
return;
}
quantity += 1;
displayQuantity(quantity);
}
(1)当满足 if 小括号内的条件时,执行大括号内的代码,否则执行 else 大括号内的代码。
(2)当 else 无需任何操作时可删去。
(3)利用 return;
提前结束 method。
(4)运算符 +=
可精简代码。
(5)用 toast 告知用户错误信息或提示,以防用户误以为 App 故障或停止运行。Google 搜索 "toast android" 进一步了解。
Intent
- 利用 Android 框架 Intent 请求其它 App 组件完成动作,使 App 能使用其他 App 的功能,避免重复开发,如相机、地图、浏览器、邮件等。Google 搜索 "common intents" 进一步了解。
- Intent 就像传球,无需指定某一个 App 接球,只要设备中有 App 响应,把球接住即可。
- Intent 的内容通常有常规动作 (Action)、动作的数据 (Data),还有 Category、Component、Extra。
- Just Java App 中的 Intent 代码如下:
Intent intent = new Intent(Intent.ACTION_SENDTO);
// only email apps should handle this
intent.setData(Uri.parse("mailto:"));
intent.putExtra(Intent.EXTRA_SUBJECT, getString(R.string.order_summary_email_subject, name));
intent.putExtra(Intent.EXTRA_TEXT, createOrderSummary(name, price, hasWhippedCream, hasChocolate));
if (intent.resolveActivity(getPackageManager()) != null) {
startActivity(intent);
}
(1)Intent的常规动作 (Aciton): ACTION_SENDTO,常量(全大写、值不变)。
(2)Intent 的动作数据 (Data): Uri.parse(“mailto:”),有标准格式,Uniform Resource Identifier。
(3)Intent 的额外数据 (Extra): EXTRA_SUBJECT、EXTRA_TEXT 分别设置邮件主题和内容的文本。
(4)if 语句判断设备中是否有 App 能响应 Intent,防止 App 崩溃。
Intent 的内容会在接下来课程中详细介绍。至此 Just Java App 的开发就告一段落了,现阶段这个咖啡订购 App 实现的功能有
- 输入订购人姓名;
- 选择订购数量,范围为 1~100;
- 选择是否添加奶盖和巧克力;
- 将订购信息(包含价格)发送到邮箱。
第一部分《布局和交互》的课程完成了,同时也迎来了入门课程的第三个实战项目:AUdacityQuiz 小测验应用,它应用了除 Intent 外已学的所有知识点,
以下是在 coding 过程中发现的几个关键点:
- 在 AndroidManifest.xml 内的 MainActivity 添加
android:windowSoftInputMode="stateHidden"
使 App 启动时输入法不会自动弹出。 - 为了防止手机在旋转后 RadioButton 的状态丢失,override 了 onSaveInstanceState 和 onRestoreInstanceState 用于保存 RadioButton 的状态。
// Save the isQuestionTwoRight and isQuestionTwoRight variable state
// When phone rotate to the landscape mode or portrait mode
@Override
public void onSaveInstanceState(Bundle savedInstanceState) {
super.onSaveInstanceState(savedInstanceState);
// Save UI state changes to the savedInstanceState.
// This bundle will be passed to onCreate if the process is killed and restarted.
savedInstanceState.putBoolean("isQuestionTwoRight", isQuestionTwoRight);
savedInstanceState.putBoolean("isQuestionThreeRight", isQuestionThreeRight);
}
@Override
public void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);
// Restore UI state from the savedInstanceState.
// This bundle has also been passed to onCreate.
isQuestionTwoRight = savedInstanceState.getBoolean("isQuestionTwoRight");
isQuestionThreeRight = savedInstanceState.getBoolean("isQuestionThreeRight");
}
- 利用 TextView 属性使两行内容的 Toast 文字居中显示。
Toast toast = Toast.makeText(this, getString(R.string.toast_line_one, result) + "\n" + getString(R.string.toast_line_two, score), Toast.LENGTH_SHORT);
// Center each of the two lines Toast texts
TextView v = toast.getView().findViewById(android.R.id.message);
if (v != null) {
v.setGravity(Gravity.CENTER);
}
toast.show();
- 在 ImageView 中通过两条指令使图片完整显示。
android:adjustViewBounds="true"
android:scaleType="centerInside"
- 通过 if/else 语句添加了一条错误提示的 Toast。
这次项目导师还给出了一些很棒的建议,例如定义 style 时应该注意的一些问题:
- 所有 style 都应该有一个 parent;
- 给 style 命名时应该遵循 parent 的命名惯性;
- 最好让自定义 style 继承自 AppCompat 而不是系统的 style,以确保能向后兼容性。