2024年Android最全手把手教你完成Android期末大作业(多功能应用型APP)(1),字节跳动今日学习内容

尾声

最后,我再重复一次,如果你想成为一个优秀的 Android 开发人员,请集中精力,对基础和重要的事情做深度研究。

对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。 整理的这些架构技术希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。

最后想要拿高薪实现技术提升薪水得到质的飞跃。最快捷的方式,就是有人可以带着你一起分析,这样学习起来最为高效,所以为了大家能够顺利进阶中高级、架构师,我特地为大家准备了一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。

当你有了学习线路,学习哪些内容,也知道以后的路怎么走了,理论看多了总要实践的。

进阶学习视频

附上:我们之前因为秋招收集的二十套一二线互联网公司Android面试真题 (含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

//建立fragment管理器

fragmentManager = getSupportFragmentManager();

//管理器开启事务,将fragment实例加入管理器

fragmentManager.beginTransaction()

.add(R.id.FragmentLayout, fragmentList.get(0), “TASK”)

.add(R.id.FragmentLayout, fragmentList.get(1), “ABSOTBED”)

.add(R.id.FragmentLayout, fragmentList.get(2), “MUSIC”)

.add(R.id.FragmentLayout, fragmentList.get(3), “WEATHER”)

.commit();

//设置fragment显示初始状态

fragmentManager.beginTransaction()

.show(fragmentList.get(1))

.hide(fragmentList.get(0))

.hide(fragmentList.get(2))

.hide(fragmentList.get(3))

.commit();

//设置底部导航栏点击选择监听事件

BottomNavigationView bottomNavigationView = findViewById(R.id.BottomNavigation);

bottomNavigationView.setOnNavigationItemSelectedListener(new BottomNavigationView.OnNavigationItemSelectedListener() {

@SuppressLint(“NonConstantResourceId”)

@Override

public boolean onNavigationItemSelected(@NonNull MenuItem item) {

// return true : show selected style

// return false: do not show

switch (item.getItemId()) {

case R.id.menu_task:

ShowFragment(0);

return true;

case R.id.menu_accounts:

ShowFragment(1);

return true;

case R.id.menu_absorbed:

ShowFragment(2);

return true;

case R.id.menu_weather:

ShowFragment(3);

return true;

default:

Log.i(TAG, “onNavigationItemSelected: Error”);

break;

}

return false;

}

});

}

public void ShowFragment(int index) {

fragmentManager.beginTransaction()

.show(fragmentList.get(index))

.hide(fragmentList.get((index + 1) % 4))

.hide(fragmentList.get((index + 2) % 4))

.hide(fragmentList.get((index + 3) % 4))

.commit();

}

二、天气显示界面


1、添加依赖(用于获取和解析天气数据)

implementation ‘com.google.code.gson:gson:2.8.6’

implementation ‘com.squareup.okhttp3:okhttp:4.9.0’

2、获取天气API接口,这里以临海市为例。使用OkHttp请求天气数据,使用Log打印测试是否能成功获取

public void RefreshWeatherData() {

OkHttpClient client = new OkHttpClient();

Request request = new Request.Builder().url(weatherUrl).build();

client.newCall(request).enqueue(new Callback() {

@Override

public void onFailure(@NonNull Call call, @NonNull IOException e) {

e.printStackTrace();

}

@Override

public void onResponse(@NonNull Call call, @NonNull Response response) throws IOException {

String weatherJson = response.body().string();

Weather weather = new Gson().fromJson(weatherJson, Weather.class);

Log.i(TAG, "onResponse: "+weatherJson);

}

});

}

3、Json数据获取成功后,根据Json数据的结构建立Weather类用于解析Json数据。

// class Weather

public class Weather {

private String city; //城市名

private String update_time; //更新时间

private List data; //每天的天气数据列表,data.get(0)为当天数据

/*

getter and setter

*/

}

// class DayData

public class DayData {

private String wea; //天气状况

private String tem; //当前温度

private String tem1; //最高温

private String tem2; //最低温

private String humidity; //湿度

private String air_level; //空气质量等级

private String air_tips; //空气质量小提示

/*

getter and setter

*/

}

4、由于OkHttp的请求是在子线程中进行的,需要使用Handler消息队列机制将解析出来的Weather实例发送到主线程用以显示在界面上。

//消息处理类

public class MyHandler extends Handler {

@Override

public void handleMessage(@NonNull Message msg) {

super.handleMessage(msg);

//what == 1 天气消息

if (msg.what == 1)

ShowWeatherInfo((Weather) msg.obj);

}

}

public void ShowWeatherInfo(Weather weather) {

String city = weather.getCity();

String wea = weather.getData().get(0).getWea();

String maxTem = weather.getData().get(0).getTem1();

String minTem = weather.getData().get(0).getTem2();

String tem = weather.getData().get(0).getTem();

String humidity = "湿度 " + weather.getData().get(0).getHumidity();

String air_level = "空气指数 " + weather.getData().get(0).getAir_level();

// tem tem1 tem2 city wea rain pm image

((TextView) findViewById(R.id.cityView)).setText(city);

((TextView) findViewById(R.id.weaView)).setText(wea);

((TextView) findViewById(R.id.mmtemView)).setText(

String.format(“%s° / %s°”, minTem.substring(0, minTem.length() - 1), maxTem.substring(0, maxTem.length() - 1)));

((TextView) findViewById(R.id.temView)).setText(tem.substring(0, tem.length() - 1) + “°”);

((TextView) findViewById(R.id.humidityView)).setText(humidity);

((TextView) findViewById(R.id.levelView)).setText(air_level);

ShowWeatherImage(wea); //根据天气状况wea显示对应的天气图片,这里不详细说明,使用switch就行

}

5、别忘了在OkHttp请求完成时发送消息

public void RefreshWeatherData() {

OkHttpClient client = new OkHttpClient();

Request request = new Request.Builder().url(weatherUrl).build();

client.newCall(request).enqueue(new Callback() {

@Override

public void onFailure(@NonNull Call call, @NonNull IOException e) {

e.printStackTrace();

}

@Override

public void onResponse(@NonNull Call call, @NonNull Response response) throws IOException {

String weatherJson = response.body().string();

Weather weather = new Gson().fromJson(weatherJson, Weather.class);

Message message = new Message();

message.what = 1;

message.obj = weather;

myHandler.sendMessage(message);

}

});

}

6、优化xml布局

三、待办事项界面


这里由于ListView是放在Fragment中的,所以直接在MainAcitivity.java中设置适配器可能会出现数据没法显示的bug。所以我直接把从数据库获取数据,Adapter的定义,ListView设置适配器的模块搬到了TaskFragment.java中。

1.在task.xml中添加ListView,先不用设置UI样式,先把数据拿到并显示在界面上

<?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”>

<TextView

android:id=“@+id/taskText”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:text=“事项”/>

<ListView

android:id=“@+id/taskListView”

android:layout_width=“match_parent”

android:layout_height=“match_parent”/>

2.创建task_item.xml布局文件(这里注意线性布局的方向及宽高,以保证task_item能放在ListView中)

<?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=“wrap_content”>

<TextView

android:id=“@+id/task_content”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/black”

android:textSize=“30dp”

android:text=“TextView” />

3.新建TaskItem类,存放事项数据

package com.example.daily.tasks;

public class TaskItem {

private int id;

private String content;

private String type;

private int status;

public TaskItem(int id, String type, String content, int status){

this.id = id;

this.type = type;

this.content = content;

this.status = status;

}

// 自行添加Get和Set方法

}

4.在TaskFragment.java中创建SQLite数据库并获取待办事项的数据

public class TaskFragment extends Fragment {

private static final String TAG = TaskFragment.class.getName();

private List taskList = new ArrayList<>();

@Nullable

@Override

public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

View view = inflater.inflate(R.layout.task, container, false);

ReadTaskDataFromSQL();

//测试数据获取是否正常

for(TaskItem item : taskList){

Log.i(TAG, "taskList “+item.getId()+” "+item.getContent());

}

return view;

}

//读取数据库并将数据存到taskList

public void ReadTaskDataFromSQL(){

MySQLiteOpenHelper openHelper = new MySQLiteOpenHelper(getActivity());

SQLiteDatabase readDatabase = openHelper.getReadableDatabase();

Cursor cursor = readDatabase.query(

“task”,

new String[]{“id”, “type”, “content”, “status”},

null,null,null,null,null

);

while(cursor.moveToNext()){

TaskItem task = new TaskItem(

cursor.getInt(0),

cursor.getString(1),

cursor.getString(2),

cursor.getInt(3)

);

taskList.add(task);

}

}

//创建SQLite数据库

public class MySQLiteOpenHelper extends SQLiteOpenHelper{

public MySQLiteOpenHelper(@Nullable Context context) {

super(context, “Daily.db”, null, 1);

}

@Override

public void onCreate(SQLiteDatabase db) {

Log.i(TAG, “onCreate: sqlite”);

//创建待办事项数据表

String create_sql =

“create table task(” +

"id INTEGER PRIMARY KEY AUTOINCREMENT, " +

"content varchar(50), " +

"type varchar(50), " +

“status int);”;

db.execSQL(create_sql);

}

@Override

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {}

}

}

5.数据获取正常以后,建立ListView适配器。这里涉及到缓存convertView的使用,使用convertView可以防止每创建一个item时就解析一个布局,这样效率肯定不高。convertView是Android提供的用于缓存的View,在第一次渲染item时,将将解析出来的View放入缓存convertView,在下一次渲染item的时候,判断convertView是否为空即可。

public class TaskAdapter extends BaseAdapter{

@Override

public int getCount() {

//测试getCount返回值是否正常

Log.i(TAG, "getCount: "+taskList.size());

return taskList.size();

}

@Override

public Object getItem(int position) {

return taskList.get(position);

}

@Override

public long getItemId(int position) {

return taskList.get(position).getId();

}

@Override

public View getView(int position, View convertView, ViewGroup parent) {

//测试getView是否执行

Log.i(TAG, "getView: "+position);

ViewHolder viewHolder;

TaskItem task = (TaskItem) getItem(position);

if(convertView == null){

viewHolder = new ViewHolder();

convertView = LayoutInflater.from(getActivity()).inflate(R.layout.task_item, null);

viewHolder.taskItemTextView = convertView.findViewById(R.id.task_content);

convertView.setTag(viewHolder);

}else{

viewHolder = (ViewHolder) convertView.getTag();

}

viewHolder.taskItemTextView.setText(task.getId()+" "+task.getContent());

return convertView;

}

}

public class ViewHolder{

TextView taskItemTextView;

}

6.在onCreateView中设置ListView的适配器

private List taskList = new ArrayList<>();

private TaskAdapter taskAdapter;

private ListView taskListView;

@Nullable

@Override

public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

View view = inflater.inflate(R.layout.task, container, false);

taskListView = view.findViewById(R.id.taskListView);

taskAdapter = new TaskAdapter();

taskListView.setAdapter(taskAdapter);

ReadTaskDataFromSQL();

return view;

}

7.设计每一条待办事项的布局样式,如图所示,布局设计就不放原码了,使用多个线性布局的嵌套,gravity,margin属性即可实现。

img:task-2.jpg

8.根据待办事项的状态显示不同按钮,并标记待办事项的重要程度。

public void ShowTaskContent(View convertView, TaskItem task){

//显示事项内容

TextView content = ((ViewHolder) convertView.getTag()).taskContent;

int status = task.getStatus();

content.setText(task.getContent());

//事项已完成 中划线 灰色

if(status == 1){

content.getPaint().setFlags(Paint.STRIKE_THRU_TEXT_FLAG);

content.setTextColor(getResources().getColor(R.color.GRAY, null));

}

//事项未完成 无中划线 黑色

if(status == 0){

content.getPaint().setFlags(0);

content.setTextColor(getResources().getColor(R.color.black, null));

}

//事项失败 无中划线 灰色

if(status == -1){

content.getPaint().setFlags(0);

content.setTextColor(getResources().getColor(R.color.GRAY, null));

}

}

public void ShowTaskLevel(View convertView, int level){

// 显示事项重要级别 level : 0~3 四个优先级 Ⅰ Ⅱ Ⅲ Ⅳ

TextView levelText = ((ViewHolder) convertView.getTag()).taskLevel;

if(level == 0){

levelText.setText(“Ⅰ”);

levelText.setTextColor(getResources().getColor(R.color.level_0, null));

}

if(level == 1){

levelText.setText(“Ⅱ”);

levelText.setTextColor(getResources().getColor(R.color.level_1, null));

}

if(level == 2){

levelText.setText(“Ⅲ”);

levelText.setTextColor(getResources().getColor(R.color.level_2, null));

}

if(level == 3){

levelText.setText(“Ⅳ”);

levelText.setTextColor(getResources().getColor(R.color.level_3, null));

}

}

9.在顶部添加五个TextView作为分类查看事项菜单,点击某一分类即可查看该分类下的所有事项,并修改被点击TextView 的样式。

/** 菜单栏模块 **/

public void SetTypeMenuOnClick(View view){

typeMenuList.add((TextView) view.findViewById(R.id.TypeMenu_default));

typeMenuList.add((TextView) view.findViewById(R.id.TypeMenu_work));

typeMenuList.add((TextView) view.findViewById(R.id.TypeMenu_study));

typeMenuList.add((TextView) view.findViewById(R.id.TypeMenu_life));

int[] color = {

getResources().getColor(R.color.defaultColor, null),

getResources().getColor(R.color.workColor, null),

getResources().getColor(R.color.studyColor, null),

getResources().getColor(R.color.lifeColor, null),

};

for(int i=0; i<4 ;i++){

int finalI = i; //分类索引值

typeMenuList.get(i).setOnClickListener(v -> {

// 点击分类的一项后设置样式

typeMenuList.get(finalI).setTextColor(Color.BLACK);

typeMenuList.get(finalI).setBackgroundColor(Color.WHITE);

typeMenuList.get((finalI+1) % 4).setBackgroundColor(color[(finalI+1) % 4]);

typeMenuList.get((finalI+1) % 4).setTextColor(Color.WHITE);

typeMenuList.get((finalI+2) % 4).setBackgroundColor(color[(finalI+2) % 4]);

typeMenuList.get((finalI+2) % 4).setTextColor(Color.WHITE);

typeMenuList.get((finalI+3) % 4).setBackgroundColor(color[(finalI+3) % 4]);

typeMenuList.get((finalI+3) % 4).setTextColor(Color.WHITE);

// 显示某一类待办数据,这里筛选taskList即可

List typeTaskList = new ArrayList<>();

String[] types = {“全部”, “工作”,“学习”,“生活”};

/* 分类索引值

0 全部

1 工作

2 学习

3 生活

*/

// 点击工作 学习 生活时分类

// TypeNow 是一个全局变量,表示当前的分类

TypeNow = types[finalI];

Log.i(TAG, "SetTypeMenuOnClick: "+TypeNow);

ReadTaskFromDatabase();

});

}

}

10.task.xml布局右上角加入一个switch控件用以隐藏已完成事项。

//隐藏已完成Switch

Switch hideCompletedTaskSwitch = view.findViewById(R.id.HideCompletedTaskView);

hideCompletedTaskSwitch.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {

@Override

public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {

if(isChecked) isHideCompleted = true;

else isHideCompleted = false;

// isHideCompleted 是一个全局变量,表示当前是否隐藏已完成事项

ReadTaskFromDatabase();

}

});

完成9,10步之后就需要修改读取数据库的模块,加入TypeNow和isHideCompleted变量加以控制。

public void ReadTaskFromDatabase(){

if (taskList.size()!=0) {

taskList.clear();

}

Cursor cursor = readDatabase.query(

“task”,

new String[]{“id”, “type”, “level”,“content”, “info”, “status”},

null,

null,

null,

null,

null

);

//隐藏,有分类

if(isHideCompleted && !TypeNow.equals(“全部”)){

//只获取未完成事项

while(cursor.moveToNext()){

if((cursor.getInt(5) == 0 ) && (cursor.getString(1).equals(TypeNow))){

TaskItem task = new TaskItem(

cursor.getInt(0),

cursor.getString(1),

cursor.getInt(2),

cursor.getString(3),

cursor.getString(4),

cursor.getInt(5)

);

taskList.add(task);

}

}

}

//不隐藏,有分类

if(!isHideCompleted && !TypeNow.equals(“全部”)){

while(cursor.moveToNext()){

if(cursor.getString(1).equals(TypeNow)){

TaskItem task = new TaskItem(

cursor.getInt(0),

cursor.getString(1),

cursor.getInt(2),

cursor.getString(3),

cursor.getString(4),

cursor.getInt(5)

);

taskList.add(task);

}

}

}

//隐藏,不分类

if(isHideCompleted && TypeNow.equals(“全部”)){

while(cursor.moveToNext()){

if(cursor.getInt(5) == 0){

TaskItem task = new TaskItem(

cursor.getInt(0),

cursor.getString(1),

cursor.getInt(2),

cursor.getString(3),

cursor.getString(4),

cursor.getInt(5)

);

taskList.add(task);

}

}

}

else{

while(cursor.moveToNext()){

TaskItem task = new TaskItem(

cursor.getInt(0),

cursor.getString(1),

cursor.getInt(2),

cursor.getString(3),

cursor.getString(4),

cursor.getInt(5)

);

taskList.add(task);

}

}

// 别忘了通知ListView适配器数据变化

taskAdapter.notifyDataSetChanged();

}

11、添加事项,这里使用的是在整个RelativeLayout布局中添加一个ImageView作为添加事项的按钮,并定义点击事件,点击时弹出对话框,在对话框中输入添加事项的信息。

自定义对话框需要先设计一个layout布局文件add_task_dialog.xml

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:layout_width=“match_parent”

android:orientation=“vertical”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:paddingBottom=“20dp”

android:paddingTop=“10dp”

android:layout_height=“wrap_content”>

<TextView

android:text=“添加事项”

android:textColor=“@color/black”

android:textSize=“25dp”

android:layout_width=“match_parent”

android:gravity=“center”

android:layout_height=“50dp”/>

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:orientation=“horizontal”>

<TextView

android:id=“@+id/textView”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:layout_marginRight=“15dp”

android:text=“事项” />

<EditText

android:id=“@+id/addTaskContentEdit”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_weight=“1”

android:ems=“10”

android:inputType=“textPersonName”

android:text=“” />

<RelativeLayout

android:layout_width=“match_parent”

android:layout_marginTop=“10dp”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:layout_height=“wrap_content”>

<LinearLayout

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:orientation=“vertical”>

<TextView

android:id=“@+id/textView2”

android:layout_width=“160dp”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:text=“事项分类” />

<RadioGroup

android:id=“@+id/typeRadioGroup”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”>

<RadioButton

android:id=“@+id/radioButton8”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/workColor”

android:text=“工作” />

<RadioButton

android:id=“@+id/radioButton7”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/studyColor”

android:text=“学习” />

<RadioButton

android:id=“@+id/radioButton6”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/lifeColor”

android:text=“生活” />

<RadioButton

android:id=“@+id/radioButton5”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/defaultColor”

android:text=“不分类” />

<LinearLayout

android:layout_width=“160dp”

android:layout_height=“wrap_content”

android:layout_alignParentEnd=“true”

android:orientation=“vertical”>

<TextView

android:id=“@+id/textView3”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:text=“重要级别” />

<RadioGroup

android:id=“@+id/levelRadioGroup”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”>

<RadioButton

android:id=“@+id/radioButton”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_0”

android:text=“0 重要且紧急” />

<RadioButton

android:id=“@+id/radioButton2”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_1”

android:text=“1 重要但不紧急” />

<RadioButton

android:id=“@+id/radioButton3”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_2”

android:text=“2 不重要但紧急” />

<RadioButton

android:id=“@+id/radioButton4”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_3”

android:text=“3 不重要且不紧急” />

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_marginTop=“10dp”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:orientation=“horizontal”>

<TextView

android:id=“@+id/textView4”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:layout_marginRight=“15dp”

android:text=“备注” />

<EditText

android:id=“@+id/addTaskInfoEdit”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:ems=“10”

android:inputType=“textPersonName”

android:text=“” />

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:gravity=“center”

android:layout_marginTop=“10dp”

android:orientation=“horizontal”>

<Button

android:id=“@+id/cancelAddButton”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginRight=“100dp”

android:text=“取消” />

<Button

android:id=“@+id/confirmAddButton”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:text=“确定” />

12、定义一个方法,实现弹出添加事项界面的对话框,并设置确认和取消按钮的点击事件,确认按钮即添加该事项到数据库并显示

public void ShowAddTaskDialog(){

//获取添加事项布局实例

View addView = getLayoutInflater().inflate(R.layout.add_task_dialog, null);

// 将该布局添加到对话框

final AlertDialog addDialog = new AlertDialog.Builder(getActivity()).setView(addView).create();

addDialog.show();

//获取对话框中的布局控件

Button cancelButton = (Button) addView.findViewById(R.id.cancelAddButton);

Button confirmButton = (Button) addView.findViewById(R.id.confirmAddButton);

EditText contentEdit = (EditText) addView.findViewById(R.id.addTaskContentEdit);

EditText infoEdit = (EditText) addView.findViewById(R.id.addTaskInfoEdit);

RadioGroup typeGroup = (RadioGroup) addView.findViewById(R.id.typeRadioGroup);

RadioGroup levelGroup = (RadioGroup) addView.findViewById(R.id.levelRadioGroup);

typeGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {

@Override

public void onCheckedChanged(RadioGroup group, int checkedId) {

}

});

levelGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {

@Override

public void onCheckedChanged(RadioGroup group, int checkedId) {

}

});

//确定按钮

confirmButton.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

// 获取输入的事项内容和备注

String addContent = contentEdit.getText().toString();

String addInfo = infoEdit.getText().toString();

//RadioGroup的选择项

RadioButton typeSelectBtn = (RadioButton) addView.findViewById(typeGroup.getCheckedRadioButtonId());

String addType = typeSelectBtn.getText().toString();

RadioButton levelSelectBtn = (RadioButton) addView.findViewById(levelGroup.getCheckedRadioButtonId());

int addLevel = Integer.parseInt(levelSelectBtn.getText().toString().substring(0,1));

//插入数据库

InsertTaskToDatabase(

new TaskItem(addType, addLevel, addContent, addInfo, 0)

);

addDialog.dismiss();

}

});

// 取消按钮

cancelButton.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

addDialog.dismiss();

}

});

}

13、然后在添加事项的点击事件中调用ShowAddTaskDialog()即可

//添加事项的按钮

ImageView addTaskImage = (ImageView) view.findViewById(R.id.addTaskImage);

addTaskImage.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

ShowAddTaskDialog();

}

});

14.长按某条事项弹出对话框,显示事项信息,可以修改,删除,标记失败。和添加事项的对话框实现原理相同,这里不详细说明,给出代码供参考

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:layout_width=“match_parent”

android:orientation=“vertical”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:paddingBottom=“20dp”

android:paddingTop=“10dp”

android:layout_height=“wrap_content”>

<TextView

android:text=“事项信息”

android:textColor=“@color/black”

android:textSize=“25dp”

android:layout_width=“match_parent”

android:gravity=“center”

android:layout_height=“50dp”/>

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:orientation=“horizontal”>

<TextView

android:id=“@+id/textView”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:layout_marginRight=“15dp”

android:text=“事项” />

<EditText

android:id=“@+id/addTaskContentEdit”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_weight=“1”

android:ems=“10”

android:inputType=“textPersonName”

android:text=“” />

<RelativeLayout

android:layout_width=“match_parent”

android:layout_marginTop=“10dp”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:layout_height=“wrap_content”>

<LinearLayout

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:orientation=“vertical”>

<TextView

android:id=“@+id/textView2”

android:layout_width=“160dp”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:text=“事项分类” />

<RadioGroup

android:id=“@+id/typeRadioGroup”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”>

<RadioButton

android:id=“@+id/workButton”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/workColor”

android:text=“工作” />

<RadioButton

android:id=“@+id/studyButton”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/studyColor”

android:text=“学习” />

<RadioButton

android:id=“@+id/lifeButton”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/lifeColor”

android:text=“生活” />

<RadioButton

android:id=“@+id/defaultButton”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/defaultColor”

android:text=“全部” />

<LinearLayout

android:layout_width=“160dp”

android:layout_height=“wrap_content”

android:layout_alignParentEnd=“true”

android:orientation=“vertical”>

<TextView

android:id=“@+id/textView3”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:text=“重要级别” />

<RadioGroup

android:id=“@+id/levelRadioGroup”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”>

<RadioButton

android:id=“@+id/level0Button”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_0”

android:text=“0 重要且紧急” />

<RadioButton

android:id=“@+id/level1Button”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_1”

android:text=“1 重要但不紧急” />

<RadioButton

android:id=“@+id/level2Button”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_2”

android:text=“2 不重要但紧急” />

<RadioButton

android:id=“@+id/level3Button”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:textColor=“@color/level_3”

android:text=“3 不重要且不紧急” />

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_marginTop=“10dp”

android:paddingLeft=“15dp”

android:paddingRight=“15dp”

android:orientation=“horizontal”>

<TextView

android:id=“@+id/textView4”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:textSize=“20dp”

android:textColor=“@color/black”

android:layout_marginRight=“15dp”

android:text=“备注” />

<EditText

android:id=“@+id/addTaskInfoEdit”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:ems=“10”

android:inputType=“textPersonName”

android:text=“” />

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:gravity=“center”

android:layout_marginTop=“20dp”

android:orientation=“horizontal”>

<LinearLayout

android:layout_width=“wrap_content”

android:layout_marginRight=“60dp”

android:orientation=“vertical”

android:layout_height=“wrap_content”>

<ImageView

android:id=“@+id/deleteTaskButton”

android:layout_width=“wrap_content”

android:layout_height=“40dp”

android:adjustViewBounds=“true”

android:src=“@drawable/delete_icon”

/>

<TextView

android:layout_width=“40dp”

android:layout_height=“wrap_content”

android:textColor=“@color/black”

android:gravity=“center”

android:textSize=“15dp”

android:layout_marginTop=“5dp”

android:text=“删除”/>

<LinearLayout

android:layout_width=“wrap_content”

android:orientation=“vertical”

android:layout_height=“wrap_content”>

<ImageView

android:id=“@+id/failTaskButton”

android:layout_width=“40dp”

android:layout_height=“40dp”

android:adjustViewBounds=“true”

android:src=“@drawable/fail_icon”

/>

<TextView

android:layout_width=“40dp”

android:layout_height=“wrap_content”

android:textColor=“#d81e06”

android:gravity=“center”

android:textSize=“15dp”

android:layout_marginTop=“5dp”

android:text=“失败”/>

<LinearLayout

android:layout_width=“wrap_content”

android:layout_marginLeft=“60dp”

android:orientation=“vertical”

android:layout_height=“wrap_content”>

<ImageView

android:id=“@+id/modifyTaskButton”

android:layout_width=“40dp”

android:layout_height=“40dp”

android:adjustViewBounds=“true”

android:src=“@drawable/modify_icon”

android:text=“修改” />

<TextView

android:layout_width=“40dp”

android:layout_height=“wrap_content”

android:textColor=“@color/purple_500”

android:gravity=“center”

android:textSize=“15dp”

android:layout_marginTop=“5dp”

android:text=“修改”/>

public void ShowTaskInfoDialog(TaskItem task){

// 获取传入的事项数据

String content = task.getContent();

String type = task.getType();

int level = task.getLevel();

String info = task.getInfo();

//获取布局

View infoView = getLayoutInflater().inflate(R.layout.task_info_dialog, null);

final AlertDialog infoDialog = new AlertDialog.Builder(getActivity()).setView(infoView).create();

infoDialog.show();

//获取对话框中的布局控件

EditText contentEdit = (EditText) infoView.findViewById(R.id.addTaskContentEdit);

EditText infoEdit = (EditText) infoView.findViewById(R.id.addTaskInfoEdit);

RadioGroup typeGroup = (RadioGroup) infoView.findViewById(R.id.typeRadioGroup);

RadioGroup levelGroup = (RadioGroup) infoView.findViewById(R.id.levelRadioGroup);

ImageView deleteImage = (ImageView) infoView.findViewById(R.id.deleteTaskButton);

ImageView modifyImage = (ImageView) infoView.findViewById(R.id.modifyTaskButton);

ImageView failImage = (ImageView) infoView.findViewById(R.id.failTaskButton);

//显示task事项信息

contentEdit.setText(content);

infoEdit.setText(info);

SetTypeRadioGroupSelected(typeGroup, type);

SetLevelRadioGroupSelected(levelGroup, level);

//删除按钮

deleteImage.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

DeleteTaskToDatabase(task);

infoDialog.dismiss();

}

});

//失败按钮

failImage.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

task.setStatus(-1);

UpDateTaskToDatabase(task);

//别忘记关闭对话框

infoDialog.dismiss();

}

});

//修改按钮

modifyImage.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

// 获取输入的事项内容和备注

String modifyContent = contentEdit.getText().toString();

String modifyInfo = infoEdit.getText().toString();

//RadioGroup的选择项

RadioButton typeSelectBtn = (RadioButton) infoView.findViewById(typeGroup.getCheckedRadioButtonId());

String modifyType = typeSelectBtn.getText().toString();

RadioButton levelSelectBtn = (RadioButton) infoView.findViewById(levelGroup.getCheckedRadioButtonId());

int modifyLevel = Integer.parseInt(levelSelectBtn.getText().toString().substring(0,1));

task.setContent(modifyContent);

task.setInfo(modifyInfo);

task.setType(modifyType);

task.setLevel(modifyLevel);

UpDateTaskToDatabase(task);

//别忘记关闭对话框

infoDialog.dismiss();

}

});

}

//在适配器的getView中,设置每条事项的长按事件:调用ShowTaskInfoDialog弹出对话框显示事项的内容

convertView.setOnLongClickListener(new View.OnLongClickListener() {

@Override

public boolean onLongClick(View v) {

ShowTaskInfoDialog(task);

return false;

}

});

四、专注计时界面

写在最后

对程序员来说,很多技术的学习都是“防御性”的。也就是说,我们是在为未来学习。我们学习新技术的目的,或是为了在新项目中应用,或仅仅是为了将来的面试。但不管怎样,一定不能“止步不前”,不能荒废掉。

![
[]


文章以下内容会给出阿里与美团的面试题(答案+解析)、面试题库、Java核心知识点梳理等

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

id.modifyTaskButton);

ImageView failImage = (ImageView) infoView.findViewById(R.id.failTaskButton);

//显示task事项信息

contentEdit.setText(content);

infoEdit.setText(info);

SetTypeRadioGroupSelected(typeGroup, type);

SetLevelRadioGroupSelected(levelGroup, level);

//删除按钮

deleteImage.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

DeleteTaskToDatabase(task);

infoDialog.dismiss();

}

});

//失败按钮

failImage.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

task.setStatus(-1);

UpDateTaskToDatabase(task);

//别忘记关闭对话框

infoDialog.dismiss();

}

});

//修改按钮

modifyImage.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

// 获取输入的事项内容和备注

String modifyContent = contentEdit.getText().toString();

String modifyInfo = infoEdit.getText().toString();

//RadioGroup的选择项

RadioButton typeSelectBtn = (RadioButton) infoView.findViewById(typeGroup.getCheckedRadioButtonId());

String modifyType = typeSelectBtn.getText().toString();

RadioButton levelSelectBtn = (RadioButton) infoView.findViewById(levelGroup.getCheckedRadioButtonId());

int modifyLevel = Integer.parseInt(levelSelectBtn.getText().toString().substring(0,1));

task.setContent(modifyContent);

task.setInfo(modifyInfo);

task.setType(modifyType);

task.setLevel(modifyLevel);

UpDateTaskToDatabase(task);

//别忘记关闭对话框

infoDialog.dismiss();

}

});

}

//在适配器的getView中,设置每条事项的长按事件:调用ShowTaskInfoDialog弹出对话框显示事项的内容

convertView.setOnLongClickListener(new View.OnLongClickListener() {

@Override

public boolean onLongClick(View v) {

ShowTaskInfoDialog(task);

return false;

}

});

四、专注计时界面

写在最后

对程序员来说,很多技术的学习都是“防御性”的。也就是说,我们是在为未来学习。我们学习新技术的目的,或是为了在新项目中应用,或仅仅是为了将来的面试。但不管怎样,一定不能“止步不前”,不能荒废掉。

[外链图片转存中…(img-mQfuzEXR-1714886831359)]

[外链图片转存中…(img-GhUV7Ab1-1714886831360)]
[]

[外链图片转存中…(img-LhLT76ne-1714886831360)]
[外链图片转存中…(img-YLuU1PF3-1714886831360)]

文章以下内容会给出阿里与美团的面试题(答案+解析)、面试题库、Java核心知识点梳理等

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 19
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android平台上使用FFmpeg需要进行交叉编译,生成适用于Android的FFmpeg库,并将其打包到apk中。以下是手把手你搭建ffmpeg命令行运行环境的步骤: 1.下载NDK 首先需要下载NDK(Native Development Kit),NDK是一个工具包,用于开发C/C++应用程序的原生库。Android Studio自带NDK,也可以从官网下载。 2.下载FFmpeg源代码 从FFmpeg的官网下载源代码,然后解压到本地。 3.配置交叉编译环境 在FFmpeg源代码根目录下创建一个build_android.sh文件,输入以下内容: ```bash #!/bin/bash NDK=$HOME/Android/Sdk/ndk-bundle # NDK路径 SYSROOT=$NDK/platforms/android-21/arch-arm/ # Android SDK路径 TOOLCHAIN=$NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-x86_64 # 工具链路径 function build_one { ./configure \ --prefix=$PREFIX \ --enable-shared \ --disable-static \ --disable-doc \ --disable-ffmpeg \ --disable-ffplay \ --disable-ffprobe \ --disable-ffserver \ --disable-debug \ --disable-network \ --disable-avdevice \ --disable-postproc \ --disable-symver \ --cross-prefix=$TOOLCHAIN/bin/arm-linux-androideabi- \ --target-os=android \ --arch=arm \ --sysroot=$SYSROOT \ --extra-cflags="-Os -fpic $ADDI_CFLAGS" \ --extra-ldflags="$ADDI_LDFLAGS" \ $ADDITIONAL_CONFIGURE_FLAG make make install } CPU=arm PREFIX=$(pwd)/android/$CPU ADDI_CFLAGS="-marm" ADDI_LDFLAGS="" build_one ``` 其中,NDK是NDK的路径,SYSROOT是Android SDK的路径,TOOLCHAIN是工具链的路径。 4.执行交叉编译命令 在终端中输入以下命令: ```bash chmod +x build_android.sh ./build_android.sh ``` 等待编译完成。编译完成后,在FFmpeg源代码根目录下会生成一个android目录,其中包含了交叉编译生成的FFmpeg库。 5.创建Android Studio项目 打开Android Studio,创建一个新项目。在app/build.gradle文件中添加以下代码: ```groovy android { compileSdkVersion 28 defaultConfig { applicationId "com.example.ffmpegdemo" minSdkVersion 21 targetSdkVersion 28 versionCode 1 versionName "1.0" testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' } } externalNativeBuild { cmake { cppFlags "" abiFilters "armeabi-v7a" arguments "-DANDROID_ARM_NEON=TRUE" } } sourceSets.main { jniLibs.srcDirs = ['src/main/jniLibs'] } ndk { abiFilters "armeabi-v7a" } } ``` 其中,externalNativeBuild和ndk是用于指定使用交叉编译生成的库的配置。 6.将FFmpeg库打包到apk中 将交叉编译生成的库复制到项目的app/src/main/jniLibs/armeabi-v7a/目录下。在app/build.gradle文件中添加以下代码: ```groovy android { sourceSets { main { jniLibs.srcDirs = ['src/main/jniLibs'] } } } ``` 然后在终端中输入以下命令: ```bash ./gradlew assembleDebug ``` 等待打包完成。打包完成后,在项目的build/outputs/apk/debug/目录下会生成一个apk文件,其中包含了FFmpeg库。 至此,就完成了搭建ffmpeg命令行运行环境的所有步骤。可以通过在MainActivity中执行FFmpeg命令来测试FFmpeg是否正常工作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值