android|开发一个简单APP的流程及记事本案例(Android Studio)

1.需求分析 (1)业务需求分析:分析潜在客户需要什么产品或服务
(2)架构分析:分析所做的产品或服务需要什么功能及实现方式并画出功能结构图,记事本案例如下
android|开发一个简单APP的流程及记事本案例(Android Studio)
文章图片

(3)数据库类设计分析:数据库设计是项目开发中非常关键的一个环节。同样在记事本案例中也至关重要,我们通过数据库表(Note)进行增删改查操作,记事本的数据表如下所示

字段名 数据类型 字段名 数据类型
id integer 编号
content text 事件内容
notetime text 保存事件的时间
(4)界面需求分析:友好的界面在移动平台开发中非常重要,也是用户使用软件的先决条件。记事本案例分为3个界面,分别为记事本界面、添加界面和修改界面
记事本界面包含添加按钮和记录列表,点击后会跳转到添加界面,界面标题为添加记录,也可在该界面清除和保存编辑的内容。当点击记事本界面中的Item时,会跳转到修改记录界面,界面标题为修改记录,在该界面中可以查看和修改已保存的记录内容,也可清除和保存编辑的内容
2.记事本案例步骤 (1)创建项目后将Activity名称设置为NotepadActivity,布局文件为activity_notepad并将所需素材add.png(添加按钮),save_note.png(保存按钮),delete.png(删除按钮),back.png(返回按钮)导入drawable文件夹中,如下
android|开发一个简单APP的流程及记事本案例(Android Studio)
文章图片
android|开发一个简单APP的流程及记事本案例(Android Studio)
文章图片
android|开发一个简单APP的流程及记事本案例(Android Studio)
文章图片
android|开发一个简单APP的流程及记事本案例(Android Studio)
文章图片

上面添加按钮右边还有个返回按钮图片是白色的
(2)编写activity_notepad.xml布局文件,放置一个TextView控件用于显示界面标题,一个ListView控件用于显示记录列表,一个ImageView控件用于显示添加按钮的图片

(3)修改清单文件
项目创建后所有界面都有一个默认的标题栏,该标题栏不太美观,因此需要在清单文件(AndroidManifest.xml)中的标签中修改android:theme属性
android:theme="@style/Theme.AppCompat.NoActionBar"

(4)搭建记事本界面Item布局
在res/layout文件夹中,创建一个布局文件notepad_item_layout.xml,在其中放置两个TextView控件,分别用来显示记录的部分内容与保存记录的时间

【android|开发一个简单APP的流程及记事本案例(Android Studio)】(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 list; public NotepadAdapter(Context context, Listlist){ 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; importjava.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 publicvoid 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 Listquery(){ List list=new ArrayList(); 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 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(); returntrue; } }); } private void showQueryData(){ if(list!=null){ list.clear(); } list=mSQLiteHelper.query(); adapter=new NotepadAdapter(this,list); listView.setAdapter(adapter); } @Override protectedvoid 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.实现修改记录界面的功能
修改记录界面主要包含查看记录和修改记录的功能
  • 实现查看记录功能
    记事本界面列表的每个Item最多只显示2行记录信息,如果想要查看更多的记录内容,则需要点击Item,进入到修改记录界面进行查看,因此点击Item时,需要将Item对应的记录信息传递到修改记录界面进行显示,即为NotepadActivity的initData()方法,具体代码已在步骤(7)中写出
    在RecordActivity的initData()方法中需要接收记事本界面传递过来的记录数据并将数据显示到界面上,具体代码已在当前步骤中写出
  • 实现修改记录功能
    在RecordActivity中的onClick()方法中,找到保存按钮的点击事件,在该事件中根据判断传递过来的id是否为null来判断处理的是添加记录界面的保存功能还是修改记录界面的保存功能,如果id不为null,则处理修改记录界面的保存功能,具体代码已在当前步骤中写出
(9)删除记事本中的记录
当需要删除记事本列表中的记录时,需要长按列表中的Item,此时会弹出一个对话框提示是否删除Item的对应的记录,因此在NotepadActivity中的initData()方法中写出了删除记录的逻辑代码,具体代码已在步骤(7)中写出
(10)记事本案例已完成

    推荐阅读