小言_互联网的博客

Android 垃圾分类APP(七)历史记录

349人阅读  评论(0)

前言

  本文将是这个垃圾分类APP的暂定最后一篇,后面可能有,可能没有,就像薛定谔的猫一样。

正文

  本文讲述垃圾分类的历史记录,为什么要这个记录呢?因为可能有时候我查询过某一件物品的分类,然后我不记得了,再查询一次我觉得麻烦,我就希望能看到以往的查询记录。这是一个很合理的要求,不是吗?

一、建表

  要保存历史数据,首先要有一个表,在上一篇文章中,我们已经建过一个News了,下面再建一个History表,在model包下新建一个History类,里面的代码如下:

package com.llw.goodtrash.model;

import org.litepal.crud.LitePalSupport;

/**
 * 历史记录实体
 *
 * @author llw
 */
public class History extends LitePalSupport {
   

    private int id;
    private String name;
    private int type;
    private int aipre;
    private String explain;
    private String contain;
    private String tip;
    private String dateTime;

    public int getId() {
   
        return id;
    }

    public void setId(int id) {
   
        this.id = id;
    }

    public String getName() {
   
        return name;
    }

    public void setName(String name) {
   
        this.name = name;
    }

    public int getType() {
   
        return type;
    }

    public void setType(int type) {
   
        this.type = type;
    }

    public int getAipre() {
   
        return aipre;
    }

    public void setAipre(int aipre) {
   
        this.aipre = aipre;
    }

    public String getExplain() {
   
        return explain;
    }

    public void setExplain(String explain) {
   
        this.explain = explain;
    }

    public String getContain() {
   
        return contain;
    }

    public void setContain(String contain) {
   
        this.contain = contain;
    }

    public String getTip() {
   
        return tip;
    }

    public void setTip(String tip) {
   
        this.tip = tip;
    }

    public String getDateTime() {
   
        return dateTime;
    }

    public void setDateTime(String dateTime) {
   
        this.dateTime = dateTime;
    }
}

然后在litepal.xml中配置一下。


注意一下,当你的数据库已经创建之后,如果要使新增的表生效,则需要数据库的版本进行升级,比如之前是1,现在我新增了一个表,那么改成2,这样拟新增的表才会生效。或者你不升级,还是1,你只要把原来的APP卸载重装就可以。

二、新增历史记录页面

在ui包下新建一个HistoryActivity,布局是activity_history.xml。下面对于这个也页面的布局还是要想一下该怎么做,首先肯定要有一个列表用来展示这个数据吧。其次要是没有数据的时候显示一片空白好像也不合适。所以还需要一个显示空数据的布局。

好的,目前先搞定这两步。先写空数据布局,在layout下新建一个layout_empty_data.xml,里面的代码如下:

<?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:gravity="center"
    android:orientation="vertical">

    <ImageView
        android:src="@mipmap/icon_empty_data"
        android:layout_width="@dimen/dp_100"
        android:layout_height="@dimen/dp_100" />

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="@dimen/dp_12"
        android:text="空空如也"
        android:textSize="@dimen/sp_16" />
</LinearLayout>

这里用到的icon_empty_data图标如下:

下面来写这个activity_history.xml代码:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    android:orientation="vertical"
    tools:context=".ui.HistoryActivity">

    <androidx.appcompat.widget.Toolbar
    	android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="?attr/actionBarSize"
        android:background="@color/white"
        app:navigationIcon="@mipmap/icon_back">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:text="历史记录"
            android:textColor="@color/black"
            android:textSize="@dimen/sp_18" />
    </androidx.appcompat.widget.Toolbar>

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_marginTop="@dimen/dp_2">

        <include
            android:id="@+id/lay_empty_data"
            layout="@layout/layout_empty_data"
            android:layout_width="match_parent"
            android:layout_height="match_parent" />
        <!--历史记录列表-->
        <androidx.recyclerview.widget.RecyclerView
            android:id="@+id/rv_history"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:visibility="gone" />
    </RelativeLayout>


</LinearLayout>

布局写好了,但是还有列表的item需要写布局和适配器。

三、列表适配器

首先写item的布局,在layout下新建item_history_rv.xml,里面的代码如下:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_marginBottom="@dimen/dp_1"
    android:background="@color/white"
    android:orientation="vertical"
    android:padding="@dimen/dp_12">

    <TextView
        android:id="@+id/tv_name"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="物品名称"
        android:textColor="@color/black"
        android:textSize="@dimen/sp_16" />

    <TextView
        android:id="@+id/tv_datetime"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentEnd="true"
        android:text="保存时间"
        android:textColor="@color/black"
        android:textSize="@dimen/sp_14" />

    <TextView
        android:id="@+id/tv_type"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_below="@+id/tv_name"
        android:layout_marginTop="@dimen/dp_8"
        android:text="垃圾类型"
        android:textColor="@color/hint_color"
        android:textSize="@dimen/sp_14" />

    <TextView
        android:id="@+id/tv_explain"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_below="@+id/tv_type"
        android:layout_marginTop="@dimen/dp_8"
        android:text="解释"
        android:textColor="@color/hint_color"
        android:textSize="@dimen/sp_14" />

</RelativeLayout>

然后就是写适配器了,在adapter包下新建一个HistoryAdapter类,里面的代码如下:

package com.llw.goodtrash.adapter;

import android.widget.TextView;

import androidx.annotation.Nullable;

import com.chad.library.adapter.base.BaseQuickAdapter;
import com.chad.library.adapter.base.BaseViewHolder;
import com.llw.goodtrash.R;
import com.llw.goodtrash.model.History;

import java.util.List;

/**
 * 历史记录列表适配器
 * @author llw
 */
public class HistoryAdapter extends BaseQuickAdapter<History, BaseViewHolder> {
   

    public HistoryAdapter(int layoutResId, @Nullable List<History> data) {
   
        super(layoutResId, data);
    }

    @Override
    protected void convert(BaseViewHolder helper, History item) {
   
        helper.setText(R.id.tv_name, item.getName())
                .setText(R.id.tv_datetime,item.getDateTime())
                .setText(R.id.tv_explain, item.getExplain());

        TextView tvType = helper.getView(R.id.tv_type);
        switch (item.getType()) {
   
            case 0:
                tvType.setText("可回收垃圾");
                break;
            case 1:
                tvType.setText("有害垃圾");
                break;
            case 2:
                tvType.setText("厨余垃圾");
                break;
            case 3:
                //干垃圾即其他垃圾
                tvType.setText("干垃圾");
                break;
            default:
                tvType.setText("可回收垃圾");
                break;
        }
    }
}

四、历史记录页面初始化

修改HistoryActivity页面代码如下:

package com.llw.goodtrash.ui;

import androidx.appcompat.widget.Toolbar;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;

import android.os.Bundle;
import android.view.View;
import android.widget.LinearLayout;

import com.llw.goodtrash.R;
import com.llw.goodtrash.adapter.HistoryAdapter;
import com.llw.goodtrash.model.History;
import com.llw.mvplibrary.base.BaseActivity;

import org.litepal.LitePal;

import java.util.List;

/**
 * 历史记录
 * @author llw
 */
public class HistoryActivity extends BaseActivity {
   

    //工具栏
    private Toolbar toolbar;
    //空数据布局
    private LinearLayout layEmptyData;
    //历史列表
    private RecyclerView rvHistory;
    //适配器
    private HistoryAdapter mAdapter;
    //历史数据列表
    private List<History> mList;

    @Override
    public void initData(Bundle savedInstanceState) {
   
        initView();
    }

    /**
     * 页面初始化
     */
    private void initView() {
   
        toolbar = findViewById(R.id.toolbar);
        //设置页面状态栏
        setStatubar(this, R.color.white, true);
        back(toolbar,false);
        layEmptyData = findViewById(R.id.lay_empty_data);
        rvHistory = findViewById(R.id.rv_history);

        //获取数据库中的历史数据
        mList = LitePal.findAll(History.class);
        if (mList.size() > 0) {
   
            //设置列表的数据
            mAdapter = new HistoryAdapter(R.layout.item_history_rv, mList);
            rvHistory.setLayoutManager(new LinearLayoutManager(context));
            rvHistory.setAdapter(mAdapter);
            layEmptyData.setVisibility(View.GONE);
            rvHistory.setVisibility(View.VISIBLE);
        } else {
   
            //隐藏列表
            layEmptyData.setVisibility(View.VISIBLE);
            rvHistory.setVisibility(View.GONE);
        }
    }

    @Override
    public int getLayoutId() {
   
        return R.layout.activity_history;
    }
}

下面我们增加一个进入历史记录页面的入口。

修改activity_main.xml。在NestedScrollView下面加上一个浮动按钮:

	<!--浮动按钮 历史记录-->
    <com.google.android.material.floatingactionbutton.FloatingActionButton
        android:id="@+id/btn_history"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="bottom|end"
        android:layout_margin="@dimen/dp_20"
        android:src="@mipmap/icon_history"
        android:onClick="jumpHistory"
        app:backgroundTint="@color/white"
        app:backgroundTintMode="screen"
        app:fabSize="mini"
        app:hoveredFocusedTranslationZ="@dimen/dp_18"
        app:pressedTranslationZ="@dimen/dp_18" />

位置如下图所示:

icon_history图标

然后在MainActivity中写一个jumpHistory方法,代码如下:

	/**
     * 进入历史记录页面
     */
    public void jumpHistory(View view) {
   
        gotoActivity(HistoryActivity.class);
    }

运行一下;

嗯,现在是空空如也,下面来添加记录,进行垃圾分类结果返回的第三有三个,分别是文字输入进行垃圾分类,语音输入进行垃圾分类,还有图像输入进行垃圾分类。下面先来看看怎么保存这个垃圾分类的信息。

五、保存历史记录

在前面的文章中当进行分类时,会关联出很多物品,而我们要保存和我输入物品的一致性才行,比如当我搜索水杯时,会出现的结果有:水杯、保温杯、汽车杯等一些物品。而我只需要保存水杯的结果到历史记录就可以了。那么在写保存方法时首先要比对这个搜索结果。一致才保存。

下面来写代码,这里我还是写一个帮助类。在utils下新建一个HistoryHelper类,里面的代码如下:

package com.llw.goodtrash.utils;

import android.util.Log;

import com.google.gson.Gson;
import com.llw.goodtrash.model.History;
import com.llw.goodtrash.model.TrashResponse;
import com.llw.mvplibrary.network.utils.DateUtil;

import org.litepal.LitePal;

import java.util.List;

/**
 * 历史记录帮助类
 *
 * @author llw
 */
public class HistoryHelper {
   

    public static final String TAG = "HistoryHelper";

    /**
     * 查询所有历史记录
     *
     * @return 结果列表
     */
    public static List<History> queryAllHistory() {
   
        return LitePal.findAll(History.class);
    }

   /**
     * 是否存在历史记录
     *
     * @param name 物品名
     * @return true or false
     */
    public static boolean isHaveHistory(String name) {
   
        List<History> histories = LitePal.where("name = ?", name).find(History.class);
        return histories.size() > 0;
    }

    /**
     * 保存历史记录
     *
     * @param list 需要保存的数据
     * @param word 物品名称
     */
    public static void saveHistory(List<TrashResponse.NewslistBean> list, String word) {
   
        for (TrashResponse.NewslistBean bean : list) {
   
            //遍历返回数据,找出返回结果中与搜索内容一致的数据,保存到数据表中
            if (bean.getName().equals(word)) {
   
                //保存数据前先查询是否存在数据
                List<History> historyList = queryAllHistory();
                //有数据则遍历检查保存
                if (historyList.size() > 0) {
   
                    if (!isHaveHistory(bean.getName())) {
   
                        //不存在则直接保存
                        saveHistory(bean);
                    } else {
   
                        Log.d(TAG, "记录已存在");
                    }
                } else {
   
                    //没有数据则直接保存
                    saveHistory(bean);
                }
            } else {
   
                Log.d(TAG, "没有匹配到相关结果,无法保存");
            }
        }

        Log.d(TAG,new Gson().toJson(queryAllHistory()));
    }

    /**
     * 保存历史
     * @param bean
     */
    private static void saveHistory(TrashResponse.NewslistBean bean) {
   
        History historyBean = new History();
        historyBean.setName(bean.getName());
        historyBean.setType(bean.getType());
        historyBean.setAipre(bean.getAipre());
        historyBean.setExplain(bean.getExplain());
        historyBean.setContain(bean.getContain());
        historyBean.setTip(bean.getTip());
        //添加历史记录的保存时间
        historyBean.setDateTime(DateUtil.getDateTime());
        historyBean.save();
        if (historyBean.save()) {
   
            Log.d(TAG, "保存历史记录成功");
        } else {
   
            Log.d(TAG, "保存历史记录失败");
        }
    }
}


下面去使用一下这个方法。
首先是文字输入页面TextInputActivity。

将之前的word变成成员变量:

private String word;//输入的物品

当点击软键盘的搜索按钮时会将输入框的内容赋值给word。

然后只要在getSearchResponse方法中保存就好了。

下面进入声音输入页面VoiceInputActivity。

private String word;//输入的物品

进行赋值

保存结果。

然后是图像输入页面ImageInputActivity。

private String word;//输入的物品

然后赋值:

最后保存。

下面基本上都有了保存,运行一下:

嗯,效果还是不错的吧。

既然有保存,那就应该有删除,理论上来说,删除也是有学问的,单项删除、多选删除、全删。而删除的方法也是多种多样的,比如点击弹窗删除,侧滑删除,编辑列表删除。各种各样的,这里我就弄一个滑动删除和全选删除吧。

六、删除历史记录

先来看看侧滑删除,这里需要用到一个第三方依赖库,打开mvplibrary下的build.gradle。
在dependencies{}闭包下添加如下依赖:

	//列表item侧滑删除
    api 'com.github.mcxtzhang:SwipeDelMenuLayout:V1.3.0'

添加位置如下:

下面修改历史列表的item布局。

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_marginBottom="@dimen/dp_1"
    android:background="@color/white">

    <!--支持侧滑的布局-->
    <com.mcxtzhang.swipemenulib.SwipeMenuLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:clickable="true"
        android:paddingBottom="1dp">
        <!--显示文本-->
        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:padding="@dimen/dp_12">

            <TextView
                android:id="@+id/tv_name"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="物品名称"
                android:textColor="@color/black"
                android:textSize="@dimen/sp_16" />

            <TextView
                android:id="@+id/tv_datetime"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_alignParentEnd="true"
                android:text="保存时间"
                android:textColor="@color/black"
                android:textSize="@dimen/sp_14" />

            <TextView
                android:id="@+id/tv_type"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_below="@+id/tv_name"
                android:layout_marginTop="@dimen/dp_8"
                android:text="垃圾类型"
                android:textColor="@color/hint_color"
                android:textSize="@dimen/sp_14" />

            <TextView
                android:id="@+id/tv_explain"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_below="@+id/tv_type"
                android:layout_marginTop="@dimen/dp_8"
                android:text="解释"
                android:textColor="@color/hint_color"
                android:textSize="@dimen/sp_14" />
        </RelativeLayout>

        <!-- 侧滑菜单的内容  删除 -->
        <Button
            android:id="@+id/btn_delete"
            android:layout_width="@dimen/dp_100"
            android:layout_height="match_parent"
            android:background="@color/red"
            android:text="删除"
            android:textColor="@color/white" />

    </com.mcxtzhang.swipemenulib.SwipeMenuLayout>


</RelativeLayout>

这里有一个red的颜色,在app模块的colors.xml中添加

<color name="red">#FF0000</color>

然后修改适配器HistoryAdapter,添加侧滑菜单的点击事件。

然后回到HistoryActivity页面去设置适配器的点击事件。

			//列表item点击事件
            mAdapter.setOnItemChildClickListener((adapter, view, position) -> {
   
                         
            });

添加位置如下图所示:

由于现在只给适配器中的一个控件设置了点击事件,因此可以直接写代码,而不需要去判断控件id了。

那么下面在HistoryHelper中添加如下两个删除方法:

	/**
     * 根据id删除数据
     * @param id id 
     */
    public static void deleteHistoryById(long id){
   
        LitePal.delete(History.class,id);
    }

    /**
     * 根据所有历史记录
     */
    public static void deleteAllHistory() {
   
        LitePal.deleteAll(History.class);
    }

下面我要修改一下HistoryActivity页面的代码:

package com.llw.goodtrash.ui;

import androidx.appcompat.widget.Toolbar;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;

import android.os.Bundle;
import android.view.View;
import android.widget.LinearLayout;

import com.chad.library.adapter.base.BaseQuickAdapter;
import com.llw.goodtrash.R;
import com.llw.goodtrash.adapter.HistoryAdapter;
import com.llw.goodtrash.model.History;
import com.llw.goodtrash.utils.HistoryHelper;
import com.llw.mvplibrary.base.BaseActivity;

import org.litepal.LitePal;

import java.util.ArrayList;
import java.util.List;

/**
 * 历史记录
 *
 * @author llw
 */
public class HistoryActivity extends BaseActivity {
   

    //工具栏
    private Toolbar toolbar;
    //空数据布局
    private LinearLayout layEmptyData;
    //历史列表
    private RecyclerView rvHistory;
    //适配器
    private HistoryAdapter mAdapter;
    //历史数据列表
    private List<History> mList = new ArrayList<>();

    @Override
    public void initData(Bundle savedInstanceState) {
   
        initView();
        showListData();
    }

    /**
     * 页面初始化
     */
    private void initView() {
   
        toolbar = findViewById(R.id.toolbar);
        //设置页面状态栏
        setStatubar(this, R.color.white, true);
        back(toolbar, false);
        layEmptyData = findViewById(R.id.lay_empty_data);
        rvHistory = findViewById(R.id.rv_history);

        mAdapter = new HistoryAdapter(R.layout.item_history_rv, mList);
        rvHistory.setLayoutManager(new LinearLayoutManager(context));
        rvHistory.setAdapter(mAdapter);
        //列表item点击事件
        mAdapter.setOnItemChildClickListener((adapter, view, position) -> {
   
            
        });
    }

    /**
     * 显示列表
     */
    private void showListData() {
   
        List<History> historyList = HistoryHelper.queryAllHistory();
        if (historyList.size() > 0) {
   
            //设置列表的数据
            mList.clear();
            mList.addAll(historyList);
            mAdapter.notifyDataSetChanged();
            layEmptyData.setVisibility(View.GONE);
            rvHistory.setVisibility(View.VISIBLE);
        } else {
   
            //隐藏列表
            layEmptyData.setVisibility(View.VISIBLE);
            rvHistory.setVisibility(View.GONE);
        }
    }
    
    @Override
    public int getLayoutId() {
   
        return R.layout.activity_history;
    }
}			

这样改写是为了方便后面的删除操作。

下面运行一下:

下面来看看全部删除,这里我们就写的简单一些,打开activity_history.xml,在toolbar控件中,增加一个全删,如下所示:

		<TextView
            android:id="@+id/tv_all_delete"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="right"
            android:onClick="deleteAll"
            android:textColor="@color/black"
            android:textSize="@dimen/sp_16"
            android:layout_marginEnd="@dimen/dp_4"
            android:text="全删"
            android:padding="@dimen/dp_12"/>


回到HistoryActivity中。

	//全删
    private TextView tvAllDelete;

绑定控件id。


控制是否显示这个按钮。

点击全删的实现代码。

	/**
     * 全删
     * @param view
     */
    public void deleteAll(View view) {
   
        HistoryHelper.deleteAllHistory();
        showListData();
    }

下面运行一下:

那么这个APP的第一版功能就差不多了,虽然很简单吧,但也是需要花点时间去做的。

再给App弄一个桌面图标吧。

如下图修改即可。

七、源码

源码地址如下:GoodTrash

CSDN资源地址:GoodTrash.rar

APK下载:GoodTrash1.0.apk

我是初学者-Study,山高水长,后会有期~


转载:https://blog.csdn.net/qq_38436214/article/details/115945625
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场