(1)业务需求分析:分析潜在客户需要什么产品或服务
(2)架构分析:分析所做的产品或服务需要什么功能及实现方式并画出功能结构图,记事本案例如下
(3)数据库类设计分析:数据库设计是项目开发中非常关键的一个环节。同样在记事本案例中也至关重要,我们通过数据库表(Note)进行增删改查操作,记事本的数据表如下所示
字段名 | 数据类型 | 字段名 | 数据类型 |
---|---|---|---|
id | integer | 是 | 编号 |
content | text | 否 | 事件内容 |
notetime | text | 否 | 保存事件的时间 |
(4)界面需求分析:友好的界面在移动平台开发中非常重要,也是用户使用软件的先决条件。记事本案例分为3个界面,分别为记事本界面、添加界面和修改界面
记事本界面包含添加按钮和记录列表,点击后会跳转到添加界面,界面标题为添加记录,也可在该界面清除和保存编辑的内容。当点击记事本界面中的Item时,会跳转到修改记录界面,界面标题为修改记录,在该界面中可以查看和修改已保存的记录内容,也可清除和保存编辑的内容
(1)创建项目后将Activity名称设置为NotepadActivity,布局文件为activity_notepad并将所需素材add.png(添加按钮),save_note.png(保存按钮),delete.png(删除按钮),back.png(返回按钮)导入drawable文件夹中,如下
上面添加按钮右边还有个返回按钮图片是白色的
(2)编写activity_notepad.xml布局文件,放置一个TextView控件用于显示界面标题,一个ListView控件用于显示记录列表,一个ImageView控件用于显示添加按钮的图片
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout 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:background="#fefefe"
tools:context=".NotepadActivity">
<TextView
android:layout_width="match_parent"
android:layout_height="45dp"
android:gravity="center"
android:text="记事本"
android:textSize="20sp"
android:textColor="@android:color/white"
android:textStyle="bold"
android:background="#fb7a6a"
android:id="@+id/note_name"
/>
<ListView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/listview"
android:cacheColorHint="#00000000"
android:divider="#B4B4B4"
android:dividerHeight="1dp"
android:fadingEdge="none"
android:listSelector="#00000000"
android:scrollbars="none"
android:layout_below="@id/note_name"
/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/add"
android:id="@+id/add"
android:layout_marginBottom="30dp"
android:layout_alignParentBottom="true"
android:layout_centerHorizontal="true"
/>
</RelativeLayout>
(3)修改清单文件
项目创建后所有界面都有一个默认的标题栏,该标题栏不太美观,因此需要在清单文件(AndroidManifest.xml)中的
标签中修改android:theme属性
android:theme="@style/Theme.AppCompat.NoActionBar"
(4)搭建记事本界面Item布局
在res/layout文件夹中,创建一个布局文件notepad_item_layout.xml,在其中放置两个TextView控件,分别用来显示记录的部分内容与保存记录的时间
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:paddingLeft="12dp">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/item_content"
android:maxLines="2"
android:ellipsize="end"
android:lineSpacingExtra="3dp"
android:paddingTop="10dp"
android:textColor="@android:color/black"
/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/item_time"
android:textColor="#fb7a6a"
android:paddingTop="5dp"
android:paddingBottom="7dp"
/>
</LinearLayout>
(5)封装记录信息实体类
由于记事本中的每个记录都会有记录内容和保存记录的时间属性,因此需要创建一个NotepadBean用于存放这些属性。选中你所创建的项目的包,右击选择New>Package,创建一个bean包,在该包中创建一个NotepadBean类,该类中定义记录信息的所有属性
package com.example.notepad.bean;
public class NotepadBean {
private String id;
private String notepadContent;
private String notepadTime;
public String getId(){
return id;
}
public void setId(String id){
this.id=id;
}
public String getNotepadContent(){
return notepadContent;
}
public void setNotepadContent(String notepadContent){
this.notepadContent=notepadContent;
}
public String getNotepadTime(){
return notepadTime;
}
public void setNotepadTime(String notepadTime){
this.notepadTime=notepadTime;
}
}
注意包名
(6)编写记事本界面列表适配器
由于记事本界面的记录列表是使用ListView控件来展示的,因此需要创建一个数据适配器NotepadAdapter对ListView控件进行数据适配。步骤如下
选中你所创建的项目的包,右击选择New>Package,创建一个adapter包,在adapter包中创建一个NotepadAdapter类继承自BaseAdapter类,并重写getCount(),getItem(),getItemId(),getView()方法,这些方法用于获取Item总数、对应Item对象、Item对象的Id、对应的Item视图,在NotepadAdapter类中创建一个ViewHolder类,在该类中初始化Item界面中的控件,具体代码如下
package com.example.notepad.adapter;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import com.example.notepad.R;
import com.example.notepad.bean.NotepadBean;
import java.util.List;
public class NotepadAdapter extends BaseAdapter {
private LayoutInflater layoutInflater;
private List<NotepadBean> list;
public NotepadAdapter(Context context, List<NotepadBean>list){
this.layoutInflater=LayoutInflater.from(context);
this.list=list;
}
@Override
public int getCount(){
return list==null?0:list.size();
}
@Override
public Object getItem(int position){
return list.get(position);
}
@Override
public long getItemId(int position){
return position;
}
@Override
public View getView(int position,View convertView, ViewGroup parent){
ViewHolder viewHolder;
if(convertView==null){
convertView=layoutInflater.inflate(R.layout.notepad_item_layout,null);
viewHolder=new ViewHolder(convertView);
convertView.setTag(viewHolder);
}else{
viewHolder=(ViewHolder)convertView.getTag();
}
NotepadBean noteInfo=(NotepadBean)getItem(position);
viewHolder.tvNoteoadContent.setText(noteInfo.getNotepadContent());
viewHolder.tvNotepadTime.setText(noteInfo.getNotepadTime());
return convertView;
}
class ViewHolder{
TextView tvNoteoadContent;
TextView tvNotepadTime;
public ViewHolder(View view){
tvNoteoadContent=(TextView)view.findViewById(R.id.item_content);
tvNotepadTime=(TextView)view.findViewById(R.id.item_time);
}
}
}
注意包名
(7)创建数据库
在记事本案例中增删改查记录的数据都是通过操作数据库完成的。因此需要创建数据库类SQLiteHelper与数据库的工具类DBUtils,步骤如下
选中你所创建的项目的包,右击选择New>Package,创建一个utils包,在utils包中创建一个DBUtils类,在该类中定义数据库的名称、表名、数据库版本、数据库表中的列名以及获取当前日期等信息
package com.example.notepad.utils;
import java.text.SimpleDateFormat;
import java.util.Date;
public class DBUtils {
public static final String DATABASE_NAME="Notepad";
public static final String DATAVASE_TABLE="Note";
public static final int DATEBASE_VERION=1;
public static final String NOTEPAD_ID="id";
public static final String NOTEPAD_CONTENT="content";
public static final String NOTEPAD_TIME="notetime";
public static final String getTime(){
SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy年MM月dd日HH:mm:ss");
Date date=new Date(System.currentTimeMillis());
return simpleDateFormat.format(date);
}
}
注意包名
选中你所创建的项目的包,右击选择New>Package,创建一个database包,在database包中创建一个SQLiteHelper类继承自SQLiteOpenHelper类
package com.example.notepad.database;
import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import com.example.notepad.bean.NotepadBean;
import com.example.notepad.utils.DBUtils;
import java.util.ArrayList;
import java.util.List;
public class SQLiteHelper extends SQLiteOpenHelper {
private SQLiteDatabase sqLiteDatabase;
public SQLiteHelper(Context context){
super(context,DBUtils.DATABASE_NAME,null,DBUtils.DATEBASE_VERION);
sqLiteDatabase=this.getWritableDatabase();
}
@Override
public void onCreate(SQLiteDatabase db){
db.execSQL("create table "+DBUtils.DATAVASE_TABLE+"( "+DBUtils.NOTEPAD_ID+" integer primary key autoincrement ,"+DBUtils.NOTEPAD_CONTENT+" text , "+DBUtils.NOTEPAD_TIME+" text ) " );
}
@Override
public void onUpgrade(SQLiteDatabase db,int oldVersion,int newVersion){}
public boolean insertData(String userContent,String userTime){
ContentValues contentValues=new ContentValues();
contentValues.put(DBUtils.NOTEPAD_CONTENT,userContent);
contentValues.put(DBUtils.NOTEPAD_TIME,userTime);
return sqLiteDatabase.insert(DBUtils.DATAVASE_TABLE,null,contentValues)>0;
}
public boolean deteleData(String id){
String sql=DBUtils.NOTEPAD_ID+"=?";
String[] contentValuesArray=new String[]{String.valueOf(id)};
return sqLiteDatabase.delete(DBUtils.DATAVASE_TABLE,sql,contentValuesArray)>0;
}
public boolean updateData(String id,String content,String userYear){
ContentValues contentValues=new ContentValues();
contentValues.put(DBUtils.NOTEPAD_CONTENT,content);
contentValues.put(DBUtils.NOTEPAD_TIME,userYear);
String sql=DBUtils.NOTEPAD_ID+"=?";
String[] strings=new String[]{id};
return sqLiteDatabase.update(DBUtils.DATAVASE_TABLE,contentValues,sql,strings)>0;
}
public List<NotepadBean>query(){
List<NotepadBean> list=new ArrayList<NotepadBean>();
Cursor cursor=sqLiteDatabase.query(DBUtils.DATAVASE_TABLE,null,null,null,null,null,DBUtils.NOTEPAD_ID+" desc");
if(cursor!=null){
while (cursor.moveToNext()){
NotepadBean noteInfo=new NotepadBean();
String id=String.valueOf(cursor.getInt(cursor.getColumnIndex(DBUtils.NOTEPAD_ID)));
String content=cursor.getString(cursor.getColumnIndex(DBUtils.NOTEPAD_CONTENT));
String time=cursor.getString(cursor.getColumnIndex(DBUtils.NOTEPAD_TIME));
noteInfo.setId(id);
noteInfo.setNotepadContent(content);
noteInfo.setNotepadTime(time);
list.add(noteInfo);
}
cursor.close();
}
return list;
}
}
注意包名
(7)实现记事本界面的显示功能
在NotepadActivity中通过创建一个showQueryData()方法查询数据库中存放的记录信息,并将该信息显示到记录列表中,同时在NotepadActivity中还实现了添加按钮的点击事件
package com.example.notepad;
import android.content.DialogInterface;
import android.content.Intent;
import android.support.v7.app.AlertDialog;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.Toast;
import com.example.notepad.adapter.NotepadAdapter;
import com.example.notepad.bean.NotepadBean;
import com.example.notepad.database.SQLiteHelper;
import java.util.List;
public class NotepadActivity extends AppCompatActivity {
ListView listView;
List<NotepadBean> list;
SQLiteHelper mSQLiteHelper;
NotepadAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_notepad);
listView=(ListView)findViewById(R.id.listview);
ImageView add=(ImageView)findViewById(R.id.add);
add.setOnClickListener(new View.OnClickListener(){
@Override
public void onClick(View v){
Intent intent=new Intent(NotepadActivity.this,RecordActivity.class);
startActivityForResult(intent,1);
}
});
initData();
}
protected void initData(){
mSQLiteHelper=new SQLiteHelper(this);
showQueryData();
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
NotepadBean notepadBean=list.get(position);
Intent intent=new Intent(NotepadActivity.this,RecordActivity.class);
intent.putExtra("id",notepadBean.getId());
intent.putExtra("time",notepadBean.getNotepadTime());
intent.putExtra("content",notepadBean.getNotepadContent());
NotepadActivity.this.startActivityForResult(intent,1);
}
});
listView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() {
@Override
public boolean onItemLongClick(AdapterView<?> parent, View view, final int position, long id) {
AlertDialog dialog;
AlertDialog.Builder builder=new AlertDialog.Builder(NotepadActivity.this).setMessage("是否删除此记录").setPositiveButton("确定", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
NotepadBean notepadBean=list.get(position);
if(mSQLiteHelper.deteleData(notepadBean.getId())){
list.remove(position);
adapter.notifyDataSetChanged();
Toast.makeText(NotepadActivity.this,"删除成功",Toast.LENGTH_SHORT).show();
}
}
}).setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
dialog.dismiss();
}
});
dialog=builder.create();
dialog.show();
return true;
}
});
}
private void showQueryData(){
if(list!=null){
list.clear();
}
list=mSQLiteHelper.query();
adapter=new NotepadAdapter(this,list);
listView.setAdapter(adapter);
}
@Override
protected void onActivityResult(int requestCode,int resultCode,Intent data){
super.onActivityResult(requestCode,resultCode,data);
if(requestCode==1&&resultCode==2){
showQueryData();;
}
}
}
注意包名
(8)搭建添加记录和修改记录界面的布局
当点击记事本界面的添加按钮时,会跳转到添加记录界面,当点击记事本界面列表中的Item时,会跳转到修改记录界面。由于这两个界面上的控件与功能基本相同,因此可以使用同一个Activity和同一个布局文件显示这两个界面,步骤如下
选中你所创建的项目的包,创建一个名为RecordActivity的Activity并将布局文件指名为activity_record,在activity_record.xml布局文件中,放置两个TextView控件,分别用于显示界面标题和记录时间,一个EditText控件用于显示输入框,三个ImageView控件分别用于显示后退键图标、删除图标以及保存按钮图标
1.实现添加记录界面的功能
由于添加记录界面的清除和保存按钮需要实现点击事件,因此将RecordActivity实现View.OnClickListener接口并重写onClick()方法,在该方法中实现将编写的记录添加到数据库中的功能
package com.example.notepad;
import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;
import com.example.notepad.database.SQLiteHelper;
import com.example.notepad.utils.DBUtils;
public class RecordActivity extends AppCompatActivity implements View.OnClickListener {
ImageView note_back;
TextView note_time;
EditText content;
ImageView deldete;
ImageView note_save;
SQLiteHelper mSQLiteHelper;
TextView noteName;
String id;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_record);
note_back = (ImageView) findViewById(R.id.note_back);
note_time = (TextView) findViewById(R.id.tv_time);
content = (EditText) findViewById(R.id.note_content);
deldete = (ImageView) findViewById(R.id.delete);
note_save = (ImageView) findViewById(R.id.note_save);
noteName = (TextView) findViewById(R.id.note_name);
note_back.setOnClickListener(this);
deldete.setOnClickListener(this);
note_save.setOnClickListener(this);
initData();
}
protected void initData() {
mSQLiteHelper = new SQLiteHelper(this);
noteName.setText("添加记录");
Intent intent = getIntent();
if (intent != null) {
id = intent.getStringExtra("id");
if (id != null) {
noteName.setText("修改记录");
content.setText(intent.getStringExtra("content"));
note_time.setText(intent.getStringExtra("time"));
note_time.setVisibility(View.VISIBLE);
}
}
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.note_back:
finish();
break;
case R.id.delete:
content.setText("");
break;
case R.id.note_save:
String noteContent = content.getText().toString().trim();
if (id != null) {
if (noteContent.length() > 0) {
if (mSQLiteHelper.updateData(id, noteContent, DBUtils.getTime())) {
showToast("修改成功");
setResult(2);
finish();
} else {
showToast("修改失败");
}
} else {
showToast("修改内容不能为空");
}
}else{
if(noteContent.length()>0){
if(mSQLiteHelper.insertData(noteContent,DBUtils.getTime())){
showToast("保存成功");
setResult(2);
finish();
}else{
showToast("保存失败");
}
}else {
showToast("修改内容不能为空");
}
}
break;
}
}
public void showToast(String message){
Toast.makeText(RecordActivity.this,message,Toast.LENGTH_SHORT).show();
}
}
注意包名
2.实现修改记录界面的功能
修改记录界面主要包含查看记录和修改记录的功能
(9)删除记事本中的记录
当需要删除记事本列表中的记录时,需要长按列表中的Item,此时会弹出一个对话框提示是否删除Item的对应的记录,因此在NotepadActivity中的initData()方法中写出了删除记录的逻辑代码,具体代码已在步骤(7)中写出
(10)记事本案例已完成