项目名称: 简约记账 项目描述:基于安卓基础知识点开发的个人记账工具,其中包括记录当天收支情况,计算当天以及当月共收支总钱数,能查看历史账单,能对于每月收支情况进行图标分析,百分比分析, 能够搜索用户关于某种记录的全部收支情况等功能。基本上能够解决个人记账的所有问题,界面简约美观,滑动流畅。 项目重要技术点: 1-绘制布局,掌握android基本view控件的属性和使用 2-熟练掌握Activity页面展示,跳转和传值 3-使用碎片fragment加载界面,滑动视图切换页面 4-自定义对话框 5-自定义软键盘绘制和使用 6-列表视图ListView以及网格视图GridView的适配器使用和页面加载 7-使用Android自带数据库,熟练创建表,并进行增删改查 8-定义drawable文件,设定布局以及控件样式 9-使用MPAndroidChart第三方框架绘制柱状图
项目亮点:
自定义软键盘
收入支出页面的滑动
点击备注弹出备注对话框,点击事件弹出时间框
功能:
1. 今日收支明细界面
为more按钮添加样式,drawable -> main_morebtn_bg
标签选shape,shape=oval代表圆形
activity_main.xml
2. 收支的每一项
item_mainlv.xml
3. 头布局的绘制
4. 记录界面
需要使用fragment
5. 自定义软键盘
键盘样式文件:xml -> key.xml
keyHeight:每一个按键高度 keyWidth:每一个按键宽度
键盘宽度:一行有四个,所以是25%;水平间距和数值间距都设置1像素
确定删除和清零设置为-3,-4,-5的原因:
创建keyboard的实体类:java -> utils - > KeyboardUtils.java
接口回调
完成点击键盘按钮然后在textview中显示对应数字的功能。
六. 收支记录页面逻辑编写
ViewPager—视图滑动切换工具——通过手势滑动可以完成View的切换,一般是用来做APP 的引导页或者实现图片轮播——意味着ViewPager中经常放的东西有两个:①图片,实现图片轮播器;②View视图,实现View 的切换,如应用启动页面。
然后我们使用的是在viewpager中存放fragment
tablayout是用来获取哪个收入/支出的哪个被选中的,依次得知需要跳转到哪个页面。
1. RecordActivity.java
ViewPager 是 android 扩展包 v4 包中的类,这个类可以让用户左右切换当前的view。是一个容器类,可以在其中添加其他的 view 类。ViewPager 需要一个 PagerAdapter 适配器类给它提供数据。经常和 Fragment 一起使用,并且提供了专门的 FragmentPagerAdapter 和 FragmentStatePagerAdapter 类供 Fragment 中的 ViewPager 使用。
ragment
一般都是viewpager和fragment联合使用
Google官方是建议我们使用Fragment来填充ViewPager的,这样 可以更加方便的生成每个Page,以及管理每个Page的生命周期!给我们提供了两个Fragment 专用的Adapter:FragmentPageAdapter(页面少+简单)和FragmentStatePagerAdapter(页面多+复杂)。
2. 创建适配器类:RecorderPagerAdapter.java
适配器作用:把fragment和viewplayer进行绑定。
3. 为主界面中的记账、搜索等按钮添加点击事件,在MainActivity.java中完成onclick方法的实现
七、建立数据库,完成点击图片变色等
1. 记录页面支出模块:outcomefragment.java
2. 新建:item_recordfrag_gv.xml,进行对支出界面上多行小图标的书写
3. 新建db包,下面新建TypeBean.class:表示收入或支出具体类型的类
4. 新建DBOpenHelper, 完成数据库操作,建表,向表中插入元素等等
1. 建了一个名为typetb的表
2. 插入表中所有所需的元素
5. 新建DBManager.java类,完成增删改查操作
6. 新建UniteApp.java类,全局应用的类,需要在AndroidManifest.xml中进行对该类的注册
7. 回到dbmanager.java写cunsor的内容
8. 回到outcomefragment.java中,定义一个名为loadDataToGV()的方法用来给gridviewer填充数据,然后需要写对应的适配器
如果是列表(单列多行形式)的使用ListView,如果是多行多列网状形式的优先使用GridView。
9. 新建TypeBase适配器,为GridView显示数据做准备
适配器进行对视图中如何显示数据的设定,如设定被选中的图片为彩色,未被选中的为灰色,设定小图标下面的文字等等。
10. 回到outcomefragment.java中,完善loadDataToGV()方法
运行后出现如下界面,但是因为还没有添加点击事件,所以点击小图片不会变颜色
11 回到outcomefragment.java中,创建setGVListener方法,添加点击事件,使得点击图片后可以变成红色,并且最上面的一行能够根据点击的图片随之变换。如点击日用品,上面也变成了日用品。
八、完成记录界面两个fragment的编写
需要完成点击“确定”按钮然后把输入的数字存储在数据库中的操作。
1. 新建public class AccountBean,描述记录一条数据的相关内容实体类。
2. outcomefragment.java重写oncreate方法,在其中初始化accountbean对象,并把初始值设置为“其他”图标的名称和图片id;
3. 在监听器中获取被点击的图形id
4. 获取时间并存入对象
5. 获取钱数
outfragment.java
boardUtils.setOnEnsureListener(new KeyboardUtils.OnEnsureListener() {
@Override
// 点击确定按钮:获取记录信息保存在数据库中;返回上一级页面;
public void onEnsure() {
// 获取输入钱数
String moneyStr = moneyEt.getText().toString();
// 如果输入的为空
if (TextUtils.isEmpty(moneyStr)||moneyStr.equals("0")) {
getActivity().finish();
return;
}
float money = Float.parseFloat(moneyStr);
// 把获取的钱数存入对象
accountBean.setMoney(money);
//获取记录的信息,保存在数据库当中
// 返回上一级页面
getActivity().finish();
}
});
6.抽取两个fragment的公共部分
把 outfragment.java改名为BaseRecordFragment,然后新建一个outfragment,让in和out都继承base的
把这里改一下:
可以完成支出和收入显示一样的布局
7. 实现支出收入显示的布局不同:(抽取公共部分,面向对象中的重写,复用,继承,防止代码冗余)
把baserecordfragment中LoadDataToGV方法中的获取数据源的部分抽取出来,然后分别在income和outcome的fragment中重写该方法;
两个里需要改的地方:kind一个为0一个为1,还有“其他”的图片一个为蓝色一个为红色
8. base中创建调用数据库的抽象方法,让;两个子类去重写该方法。
八、收支界面插入数据库
1. DBOpenHelper中写建表语句
//创建记账表accounttb
sql = "create table accounttb(id integer primary key autoincrement,typename varchar(10),sImageId integer,beizhu varchar(80),money float," +
"time varchar(60),year integer,month integer,day integer,kind integer)";
db.execSQL(sql);
2. DBManager中存入数据库
/*
* 向记账表当中插入一条元素
* */
public static void insertItemToAccounttb(AccountBean bean){
// 把所有数据都存入value中
ContentValues values = new ContentValues();
values.put("typename",bean.getTypename());
values.put("sImageId",bean.getsImageId());
values.put("beizhu",bean.getBeizhu());
values.put("money",bean.getMoney());
values.put("time",bean.getTime());
values.put("year",bean.getYear());
values.put("month",bean.getMonth());
values.put("day",bean.getDay());
values.put("kind",bean.getKind());
// 插入
db.insert("accounttb",null,values);
}
3. 在OutcomeFragment中执行插入数据的方法。同理Income也是。
@Override
public void saveAccountToDB() {
// 设置收入还是支出
accountBean.setKind(0);
// 插入一条数据
DBManager.insertItemToAccounttb(accountBean);
}
如何点击备注然后弹出备注对话框?
1. 先写备注对话框的布局:dialog_beizhu.xml
2. 把对话框封装在类里:utils -> BeiZhuDialog.java,在其中使用了接口回调的方式
3. 让basefragment实现onclicklistener接口,添加onclick方法和对应的showBZDialog
4. BeiZhuDialog.java设置点击备注后自动弹出软键盘以及让对话框大小和窗口匹配的功能。
如何点击时间然后弹出日历对话框?
1. 先写对话框的布局:dialog_time.xml
2. 把对话框封装在类里:utils -> SelectTimeDialog.java,在其中使用了接口回调的方式
3. 让basefragment的onclick方法中添加showTimeDialog方法
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.frag_record_tv_time:
showTimeDialog();
break;
case R.id.frag_record_tv_beizhu:
showBZDialog();
break;
}
}
4. SelectTimeDialog.java写onclick点击事件的具体内容
5. basefragment中完善showTimeDialog
/* 弹出显示时间的对话框*/
private void showTimeDialog() {
SelectTimeDialog dialog = new SelectTimeDialog(getContext());
dialog.show();
//设定确定按钮被点击了的监听器
dialog.setOnEnsureListener(new SelectTimeDialog.OnEnsureListener() {
@Override
public void onEnsure(String time, int year, int month, int day) {
timeTv.setText(time);
accountBean.setTime(time);
accountBean.setYear(year);
accountBean.setMonth(month);
accountBean.setDay(day);
}
});
}
简约记账主页面:显示当天记录的每条信息
要把已经写好的item_mainlv现实中在主界面上
1. MainActivity.java文件中找到需要展示的listview并且生命数据源
2. 设置适配器 accountadapter.java
3. 在mainactivity.java中配置适配器
3. 去DBManager.java中取当天的所有数据
/*
* 获取记账表当中某一月的所有支出或者收入情况
* */
public static List<AccountBean>getAccountListOneMonthFromAccounttb(int year,int month){
List<AccountBean>list = new ArrayList<>();
String sql = "select * from accounttb where year=? and month=? order by id desc";
Cursor cursor = db.rawQuery(sql, new String[]{year + "", month + ""});
//遍历符合要求的每一行数据
while (cursor.moveToNext()) {
int id = cursor.getInt(cursor.getColumnIndexOrThrow("id"));
String typename = cursor.getString(cursor.getColumnIndexOrThrow("typename"));
String beizhu = cursor.getString(cursor.getColumnIndexOrThrow("beizhu"));
String time = cursor.getString(cursor.getColumnIndexOrThrow("time"));
int sImageId = cursor.getInt(cursor.getColumnIndexOrThrow("sImageId"));
int kind = cursor.getInt(cursor.getColumnIndexOrThrow("kind"));
float money = cursor.getFloat(cursor.getColumnIndexOrThrow("money"));
int day = cursor.getInt(cursor.getColumnIndexOrThrow("day"));
AccountBean accountBean = new AccountBean(id, typename, sImageId, beizhu, money, time, year, month, day, kind);
list.add(accountBean);
}
return list;
}
4. 在MainActivity.java中写inittime方法
5. 在MainActivity.java中写onresume方法,里面调用loadDBdata方法,加载数据库数据
private void loadDBData() {
// 数据库中查到的今天的信息
List<AccountBean> list = DBManager.getAccountListOneDayFromAccounttb(year, month, day);
mDatas.clear();
// 把今天的信息添加进来
mDatas.addAll(list);
// 数据库更新
adapter.notifyDataSetChanged();
}
给今日收支界面添加如下头布局
1. 添加头布局方法 addLVHeaderView();
2. 新建initview方法,初始化视图,显示除了头布局的页面(修改的)
3. mainactivity.java中onclick方法设置对应点击事件
显示头布局的金额
此时数据还都是0,需要让头布局显示本月支出和收入以及今日支出和收入的数据
1. 在DBManager中插入数据
/**
* 获取某一天的支出或者收入的总金额 kind:支出==0 收入===1
* */
public static float getSumMoneyOneDay(int year,int month,int day,int kind){
float total = 0.0f;
String sql = "select sum(money) from accounttb where year=? and month=? and day=? and kind=?";
Cursor cursor = db.rawQuery(sql, new String[]{year + "", month + "", day + "", kind + ""});
// 遍历
if (cursor.moveToFirst()) {
float money = cursor.getFloat(cursor.getColumnIndexOrThrow("sum(money)"));
total = money;
}
cursor.close();
return total;
}
2. 在mainactivity中写 setTopTvShow()方法,在onresume方法中调用
点击眼睛:设置明文密文
在mainactivity中写toggleshow的方法,在onclick中调用
点击预算剩余可以设置预算,并且进行实时展示
1. 先写设置预算对话框的布局:dialog_budget.xml
2. 新建utils->BudgetDialog.java,设定点击事件,回调接口等,
3. 在mainactivity中新建共享数据类型,保存预算数据: SharedPreferences preference
4. 在mainactivity的onclick中声明显示对话框showBudgetDialog()的方法,并且完成该方法
关于接口回调啥的可以看这个研究
这是在另一个文件中涉及到接口的部分
5. 在mainactivity中完善setTopTvShow方法中的计算预算剩余部分,如果不写,那新建一笔收入或支出后预算剩余不会更新,还是会显示之前的预算剩余,这里是为了完成记一笔之后剩余的更新。