|
|
|
@ -1,996 +0,0 @@
|
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2010-2011, The MiCode Open Source Community (www.micode.net)
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
package net.micode.notes.ui;
|
|
|
|
|
|
|
|
|
|
import android.app.Activity;
|
|
|
|
|
import android.app.AlertDialog;
|
|
|
|
|
import android.app.Dialog;
|
|
|
|
|
import android.appwidget.AppWidgetManager;
|
|
|
|
|
import android.content.AsyncQueryHandler;
|
|
|
|
|
import android.content.ContentResolver;
|
|
|
|
|
import android.content.ContentValues;
|
|
|
|
|
import android.content.Context;
|
|
|
|
|
import android.content.DialogInterface;
|
|
|
|
|
import android.content.Intent;
|
|
|
|
|
import android.content.SharedPreferences;
|
|
|
|
|
import android.database.Cursor;
|
|
|
|
|
import android.os.AsyncTask;
|
|
|
|
|
import android.os.Bundle;
|
|
|
|
|
import android.preference.PreferenceManager;
|
|
|
|
|
import android.text.Editable;
|
|
|
|
|
import android.text.TextUtils;
|
|
|
|
|
import android.text.TextWatcher;
|
|
|
|
|
import android.util.Log;
|
|
|
|
|
import android.view.ActionMode;
|
|
|
|
|
import android.view.ContextMenu;
|
|
|
|
|
import android.view.ContextMenu.ContextMenuInfo;
|
|
|
|
|
import android.view.Display;
|
|
|
|
|
import android.view.HapticFeedbackConstants;
|
|
|
|
|
import android.view.LayoutInflater;
|
|
|
|
|
import android.view.Menu;
|
|
|
|
|
import android.view.MenuItem;
|
|
|
|
|
import android.view.MenuItem.OnMenuItemClickListener;
|
|
|
|
|
import android.view.MotionEvent;
|
|
|
|
|
import android.view.View;
|
|
|
|
|
import android.view.View.OnClickListener;
|
|
|
|
|
import android.view.View.OnCreateContextMenuListener;
|
|
|
|
|
import android.view.View.OnTouchListener;
|
|
|
|
|
import android.view.inputmethod.InputMethodManager;
|
|
|
|
|
import android.widget.AdapterView;
|
|
|
|
|
import android.widget.AdapterView.OnItemClickListener;
|
|
|
|
|
import android.widget.AdapterView.OnItemLongClickListener;
|
|
|
|
|
import android.widget.Button;
|
|
|
|
|
import android.widget.EditText;
|
|
|
|
|
import android.widget.ListView;
|
|
|
|
|
import android.widget.PopupMenu;
|
|
|
|
|
import android.widget.TextView;
|
|
|
|
|
import android.widget.Toast;
|
|
|
|
|
|
|
|
|
|
import net.micode.notes.R;
|
|
|
|
|
import net.micode.notes.data.Notes;
|
|
|
|
|
import net.micode.notes.data.Notes.NoteColumns;
|
|
|
|
|
import net.micode.notes.gtask.remote.GTaskSyncService;
|
|
|
|
|
import net.micode.notes.model.WorkingNote;
|
|
|
|
|
import net.micode.notes.tool.BackupUtils;
|
|
|
|
|
import net.micode.notes.tool.DataUtils;
|
|
|
|
|
import net.micode.notes.tool.ResourceParser;
|
|
|
|
|
import net.micode.notes.ui.NotesListAdapter.AppWidgetAttribute;
|
|
|
|
|
import net.micode.notes.widget.NoteWidgetProvider_2x;
|
|
|
|
|
import net.micode.notes.widget.NoteWidgetProvider_4x;
|
|
|
|
|
|
|
|
|
|
import java.io.BufferedReader;
|
|
|
|
|
import java.io.IOException;
|
|
|
|
|
import java.io.InputStream;
|
|
|
|
|
import java.io.InputStreamReader;
|
|
|
|
|
import java.util.HashSet;
|
|
|
|
|
|
|
|
|
|
public class NotesListActivity extends Activity implements OnClickListener, OnItemLongClickListener {
|
|
|
|
|
// 定义了一些常量,例如查询标记、菜单标识等。
|
|
|
|
|
private static final int FOLDER_NOTE_LIST_QUERY_TOKEN = 0;
|
|
|
|
|
private static final int FOLDER_LIST_QUERY_TOKEN = 1;
|
|
|
|
|
private static final int MENU_FOLDER_DELETE = 0;
|
|
|
|
|
private static final int MENU_FOLDER_VIEW = 1;
|
|
|
|
|
private static final int MENU_FOLDER_CHANGE_NAME = 2;
|
|
|
|
|
|
|
|
|
|
// 定义了一些配置选项的键值对,例如是否显示添加笔记的引导。
|
|
|
|
|
private static final String PREFERENCE_ADD_INTRODUCTION = "net.micode.notes.introduction";
|
|
|
|
|
|
|
|
|
|
// 定义了一个枚举类型,表示列表编辑状态,包括笔记列表、子文件夹、通话记录文件夹。
|
|
|
|
|
private enum ListEditState {
|
|
|
|
|
NOTE_LIST, SUB_FOLDER, CALL_RECORD_FOLDER
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// 初始化背景查询处理类,用于异步查询数据库。
|
|
|
|
|
private BackgroundQueryHandler mBackgroundQueryHandler;
|
|
|
|
|
|
|
|
|
|
// 初始化 NotesListAdapter,它是用来管理列表视图中的项目适配器。
|
|
|
|
|
private NotesListAdapter mNotesListAdapter;
|
|
|
|
|
|
|
|
|
|
// 初始化 ListView,它是显示笔记列表的控件。
|
|
|
|
|
private ListView mNotesListView;
|
|
|
|
|
|
|
|
|
|
// 初始化添加新笔记的按钮。
|
|
|
|
|
private Button mAddNewNote;
|
|
|
|
|
|
|
|
|
|
// 一些标志位和变量,用于处理列表的触摸和滚动事件。
|
|
|
|
|
private boolean mDispatch;
|
|
|
|
|
private int mOriginY;
|
|
|
|
|
private int mDispatchY;
|
|
|
|
|
|
|
|
|
|
// 初始化标题栏的 TextView。
|
|
|
|
|
private TextView mTitleBar;
|
|
|
|
|
|
|
|
|
|
// 初始化当前文件夹的ID。
|
|
|
|
|
private long mCurrentFolderId;
|
|
|
|
|
|
|
|
|
|
// 初始化内容解析器,用于与数据库交互。
|
|
|
|
|
private ContentResolver mContentResolver;
|
|
|
|
|
|
|
|
|
|
// 初始化模式回调接口,用于处理多选模式下的交互。
|
|
|
|
|
private ModeCallback mModeCallBack;
|
|
|
|
|
|
|
|
|
|
// 定义了 log 标签,用于调试。
|
|
|
|
|
private static final String TAG = "NotesListActivity";
|
|
|
|
|
|
|
|
|
|
// 定义了一些常量,例如笔记列表滚动速度等。
|
|
|
|
|
public static final int NOTES_LISTVIEW_SCROLL_RATE = 30;
|
|
|
|
|
|
|
|
|
|
// 初始化焦点笔记数据项,用于管理当前选中的笔记。
|
|
|
|
|
private NoteItemData mFocusNoteDataItem;
|
|
|
|
|
|
|
|
|
|
// 定义了一些选择器字符串,用于数据库查询。
|
|
|
|
|
private static final String NORMAL_SELECTION = NoteColumns.PARENT_ID + "=?";
|
|
|
|
|
private static final String ROOT_FOLDER_SELECTION = "(" + NoteColumns.TYPE + "<>"
|
|
|
|
|
+ Notes.TYPE_SYSTEM + " AND " + NoteColumns.PARENT_ID + "=?)" + " OR ("
|
|
|
|
|
+ NoteColumns.ID + "=" + Notes.ID_CALL_RECORD_FOLDER + " AND "
|
|
|
|
|
+ NoteColumns.NOTES_COUNT + ">0)";
|
|
|
|
|
|
|
|
|
|
// 定义了一些请求码,用于区分不同的操作。
|
|
|
|
|
private final static int REQUEST_CODE_OPEN_NODE = 102;
|
|
|
|
|
private final static int REQUEST_CODE_NEW_NODE = 103;
|
|
|
|
|
|
|
|
|
|
// onCreate 方法是 Activity 生命周期中的一个回调方法,用于在 Activity 创建时进行初始化。
|
|
|
|
|
@Override
|
|
|
|
|
protected void onCreate(Bundle savedInstanceState) {
|
|
|
|
|
super.onCreate(savedInstanceState);
|
|
|
|
|
setContentView(R.layout.note_list); // 设置界面的布局文件。
|
|
|
|
|
initResources(); // 初始化资源,如布局中的控件。
|
|
|
|
|
|
|
|
|
|
// 插入应用信息的引导,当用户第一次使用应用时显示。
|
|
|
|
|
setAppInfoFromRawRes();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onActivityResult 方法是 Activity 生命周期中的一个回调方法,用于处理活动的结果。
|
|
|
|
|
@Override
|
|
|
|
|
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
|
|
|
if (resultCode == RESULT_OK
|
|
|
|
|
&& (requestCode == REQUEST_CODE_OPEN_NODE || requestCode == REQUEST_CODE_NEW_NODE)) {
|
|
|
|
|
// 如果 activity 结果是成功的,并且请求码是打开节点或新建节点,则刷新笔记列表适配器。
|
|
|
|
|
mNotesListAdapter.changeCursor(null);
|
|
|
|
|
} else {
|
|
|
|
|
// 否则,调用父类的 onActivityResult 方法处理其他情况。
|
|
|
|
|
super.onActivityResult(requestCode, resultCode, data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// setAppInfoFromRawRes 方法用于从资源文件中读取应用信息,通常用于首次启动应用时显示引导信息。
|
|
|
|
|
private void setAppInfoFromRawRes() {
|
|
|
|
|
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this);
|
|
|
|
|
// 检查是否已经显示过添加笔记的引导。
|
|
|
|
|
if (!sp.getBoolean(PREFERENCE_ADD_INTRODUCTION, false)) {
|
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
|
InputStream in = null;
|
|
|
|
|
try {
|
|
|
|
|
// 尝试打开资源文件夹中的引导信息文件。
|
|
|
|
|
in = getResources().openRawResource(R.raw.introduction);
|
|
|
|
|
if (in != null) {
|
|
|
|
|
// 如果文件打开成功,则读取内容。
|
|
|
|
|
InputStreamReader isr = new InputStreamReader(in);
|
|
|
|
|
BufferedReader br = new BufferedReader(isr);
|
|
|
|
|
char [] buf = new char[1024];
|
|
|
|
|
int len = 0;
|
|
|
|
|
while ((len = br.read(buf)) > 0) {
|
|
|
|
|
sb.append(buf, 0, len);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// 如果文件打开失败,记录错误。
|
|
|
|
|
Log.e(TAG, "Read introduction file error");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} catch (IOException e) {
|
|
|
|
|
// 如果读取文件过程中发生异常,记录异常。
|
|
|
|
|
e.printStackTrace();
|
|
|
|
|
return;
|
|
|
|
|
} finally {
|
|
|
|
|
// 确保流关闭,防止资源泄露。
|
|
|
|
|
if(in != null) {
|
|
|
|
|
try {
|
|
|
|
|
in.close();
|
|
|
|
|
} catch (IOException e) {
|
|
|
|
|
// TODO Auto-generated catch block
|
|
|
|
|
e.printStackTrace();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 使用读取的内容创建一个空白笔记。
|
|
|
|
|
WorkingNote note = WorkingNote.createEmptyNote(this, Notes.ID_ROOT_FOLDER,
|
|
|
|
|
AppWidgetManager.INVALID_APPWIDGET_ID, Notes.TYPE_WIDGET_INVALIDE,
|
|
|
|
|
ResourceParser.RED);
|
|
|
|
|
// 设置笔记的内容为读取的文本。
|
|
|
|
|
note.setWorkingText(sb.toString());
|
|
|
|
|
// 尝试保存笔记,如果成功,则标记为已显示引导信息。
|
|
|
|
|
if (note.saveNote()) {
|
|
|
|
|
sp.edit().putBoolean(PREFERENCE_ADD_INTRODUCTION, true).commit();
|
|
|
|
|
} else {
|
|
|
|
|
// 如果保存笔记失败,记录错误。
|
|
|
|
|
Log.e(TAG, "Save introduction note error");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onStart 方法是 Activity 生命周期中的一个回调方法,在 Activity 启动时调用。
|
|
|
|
|
@Override
|
|
|
|
|
protected void onStart() {
|
|
|
|
|
super.onStart(); // 调用父类的 onStart 方法。
|
|
|
|
|
startAsyncNotesListQuery(); // 启动异步的笔记列表查询。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// initResources 方法用于初始化应用的资源,如 ContentResolver、查询处理器、当前文件夹 ID 等。
|
|
|
|
|
private void initResources() {
|
|
|
|
|
mContentResolver = this.getContentResolver(); // 获取 ContentResolver。
|
|
|
|
|
mBackgroundQueryHandler = new BackgroundQueryHandler(this.getContentResolver()); // 创建一个后台查询处理器。
|
|
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER; // 设置当前文件夹 ID 为根文件夹。
|
|
|
|
|
mNotesListView = (ListView) findViewById(R.id.notes_list); // 获取笔记列表视图。
|
|
|
|
|
mNotesListView.addFooterView(LayoutInflater.from(this).inflate(R.layout.note_list_footer, null), null, false); // 向列表视图添加一个脚部视图。
|
|
|
|
|
mNotesListView.setOnItemClickListener(new OnListItemClickListener()); // 设置列表项点击监听器。
|
|
|
|
|
mNotesListView.setOnItemLongClickListener(this); // 设置列表项长按监听器。
|
|
|
|
|
mNotesListAdapter = new NotesListAdapter(this); // 创建一个 NotesListAdapter。
|
|
|
|
|
mNotesListView.setAdapter(mNotesListAdapter); // 为列表视图设置适配器。
|
|
|
|
|
mAddNewNote = (Button) findViewById(R.id.btn_new_note); // 获取添加新笔记的按钮。
|
|
|
|
|
mAddNewNote.setOnClickListener(this); // 设置按钮点击监听器。
|
|
|
|
|
mAddNewNote.setOnTouchListener(new NewNoteOnTouchListener()); // 设置按钮触摸监听器。
|
|
|
|
|
mDispatch = false; // 初始化一些标志位和变量。
|
|
|
|
|
mDispatchY = 0;
|
|
|
|
|
mOriginY = 0;
|
|
|
|
|
mTitleBar = (TextView) findViewById(R.id.tv_title_bar); // 获取标题栏的 TextView。
|
|
|
|
|
mState = ListEditState.NOTE_LIST; // 设置列表编辑状态为笔记列表。
|
|
|
|
|
mModeCallBack = new ModeCallback(); // 创建一个 ModeCallback 实例。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ModeCallback 类实现了 ListView.MultiChoiceModeListener 和 OnMenuItemClickListener 接口,用于处理列表的多选模式和菜单项点击事件。
|
|
|
|
|
private class ModeCallback implements ListView.MultiChoiceModeListener, OnMenuItemClickListener {
|
|
|
|
|
private DropdownMenu mDropDownMenu; // 定义一个 DropdownMenu 对象。
|
|
|
|
|
private ActionMode mActionMode; // 定义一个 ActionMode 对象。
|
|
|
|
|
private MenuItem mMoveMenu; // 定义一个菜单项,用于移动操作。
|
|
|
|
|
|
|
|
|
|
// onCreateActionMode 方法在 ActionMode 创建时调用,用于初始化 ActionMode。
|
|
|
|
|
public boolean onCreateActionMode(ActionMode mode, Menu menu) {
|
|
|
|
|
getMenuInflater().inflate(R.menu.note_list_options, menu); // 加载菜单资源。
|
|
|
|
|
menu.findItem(R.id.delete).setOnMenuItemClickListener(this); // 设置删除菜单项的点击监听器。
|
|
|
|
|
mMoveMenu = menu.findItem(R.id.move); // 获取移动菜单项。
|
|
|
|
|
if (mFocusNoteDataItem.getParentId() == Notes.ID_CALL_RECORD_FOLDER
|
|
|
|
|
|| DataUtils.getUserFolderCount(mContentResolver) == 0) {
|
|
|
|
|
mMoveMenu.setVisible(false); // 如果焦点项是通话记录文件夹或没有用户文件夹,则隐藏移动菜单项。
|
|
|
|
|
} else {
|
|
|
|
|
mMoveMenu.setVisible(true); // 否则,显示移动菜单项。
|
|
|
|
|
mMoveMenu.setOnMenuItemClickListener(this); // 设置移动菜单项的点击监听器。
|
|
|
|
|
}
|
|
|
|
|
mActionMode = mode; // 保存 ActionMode 对象。
|
|
|
|
|
mNotesListAdapter.setChoiceMode(true); // 设置适配器为多选模式。
|
|
|
|
|
mNotesListView.setLongClickable(false); // 设置列表项不可长按。
|
|
|
|
|
mAddNewNote.setVisibility(View.GONE); // 隐藏添加新笔记的按钮。
|
|
|
|
|
|
|
|
|
|
View customView = LayoutInflater.from(NotesListActivity.this).inflate(
|
|
|
|
|
R.layout.note_list_dropdown_menu, null); // 加载自定义视图。
|
|
|
|
|
mode.setCustomView(customView); // 设置自定义视图到 ActionMode。
|
|
|
|
|
mDropDownMenu = new DropdownMenu(NotesListActivity.this,
|
|
|
|
|
(Button) customView.findViewById(R.id.selection_menu), R.menu.note_list_dropdown); // 创建一个 DropdownMenu。
|
|
|
|
|
mDropDownMenu.setOnDropdownMenuItemClickListener(new PopupMenu.OnMenuItemClickListener() { // 设置下拉菜单项点击监听器。
|
|
|
|
|
public boolean onMenuItemClick(MenuItem item) {
|
|
|
|
|
mNotesListAdapter.selectAll(!mNotesListAdapter.isAllSelected()); // 选择或取消所有列表项。
|
|
|
|
|
updateMenu(); // 更新菜单状态。
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// updateMenu 方法用于更新下拉菜单的标题,显示当前选中项目的数量。
|
|
|
|
|
private void updateMenu() {
|
|
|
|
|
int selectedCount = mNotesListAdapter.getSelectedCount(); // 获取选中项目的数量。
|
|
|
|
|
// 更新下拉菜单
|
|
|
|
|
String format = getResources().getString(R.string.menu_select_title, selectedCount);
|
|
|
|
|
mDropDownMenu.setTitle(format); // 设置下拉菜单的标题。
|
|
|
|
|
MenuItem item = mDropDownMenu.findItem(R.id.action_select_all); // 获取全选菜单项。
|
|
|
|
|
if (item != null) { // 检查菜单项是否存在。
|
|
|
|
|
if (mNotesListAdapter.isAllSelected()) { // 检查是否已经全选。
|
|
|
|
|
item.setChecked(true); // 如果全选,则设置菜单项为选中状态。
|
|
|
|
|
item.setTitle(R.string.menu_deselect_all); // 设置菜单项标题为取消全选。
|
|
|
|
|
} else {
|
|
|
|
|
item.setChecked(false); // 否则,设置菜单项为未选中状态。
|
|
|
|
|
item.setTitle(R.string.menu_select_all); // 设置菜单项标题为全选。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onPrepareActionMode 方法在 ActionMode 准备时调用,可以用来准备菜单等。
|
|
|
|
|
public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onActionItemClicked 方法在用户点击菜单项时调用,可以在这里处理菜单项的点击事件。
|
|
|
|
|
public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onDestroyActionMode 方法在 ActionMode 销毁时调用,可以在这里清理状态。
|
|
|
|
|
public void onDestroyActionMode(ActionMode mode) {
|
|
|
|
|
mNotesListAdapter.setChoiceMode(false); // 关闭多选模式。
|
|
|
|
|
mNotesListView.setLongClickable(true); // 恢复列表项的长按可点击性。
|
|
|
|
|
mAddNewNote.setVisibility(View.VISIBLE); // 显示添加新笔记的按钮。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// finishActionMode 方法用于结束当前的多选模式。
|
|
|
|
|
public void finishActionMode() {
|
|
|
|
|
mActionMode.finish(); // 结束 ActionMode。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onItemCheckedStateChanged 方法在用户选中或取消选中列表项时调用,可以在这里更新状态。
|
|
|
|
|
public void onItemCheckedStateChanged(ActionMode mode, int position, long id,
|
|
|
|
|
boolean checked) {
|
|
|
|
|
mNotesListAdapter.setCheckedItem(position, checked); // 更新适配器中的选中状态。
|
|
|
|
|
updateMenu(); // 更新菜单状态。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onMenuItemClick 方法用于处理菜单项的点击事件,可以在这里添加点击事件的逻辑。
|
|
|
|
|
public boolean onMenuItemClick(MenuItem item) {
|
|
|
|
|
if (mNotesListAdapter.getSelectedCount() == 0) { // 如果没有选中任何项目。
|
|
|
|
|
Toast.makeText(NotesListActivity.this, getString(R.string.menu_select_none),
|
|
|
|
|
Toast.LENGTH_SHORT).show(); // 显示提示信息。
|
|
|
|
|
return true; // 返回 true 表示处理了该事件。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (item.getItemId()) { // 处理菜单项的点击事件。
|
|
|
|
|
case R.id.delete: // 如果点击了删除菜单项。
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
builder.setTitle(getString(R.string.alert_title_delete));
|
|
|
|
|
builder.setIcon(android.R.drawable.ic_dialog_alert);
|
|
|
|
|
builder.setMessage(getString(R.string.alert_message_delete_notes,
|
|
|
|
|
mNotesListAdapter.getSelectedCount()));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok,
|
|
|
|
|
new DialogInterface.OnClickListener() {
|
|
|
|
|
public void onClick(DialogInterface dialog,
|
|
|
|
|
int which) {
|
|
|
|
|
batchDelete(); // 调用批量删除方法。
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
builder.setNegativeButton(android.R.string.cancel, null);
|
|
|
|
|
builder.show(); // 显示确认删除对话框。
|
|
|
|
|
break;
|
|
|
|
|
case R.id.move: // 如果点击了移动菜单项。
|
|
|
|
|
startQueryDestinationFolders(); // 启动查询目的地文件夹的方法。
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return false; // 默认返回 false,表示不处理该菜单项的点击事件。
|
|
|
|
|
}
|
|
|
|
|
return true; // 返回 true,表示处理了该事件。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewNoteOnTouchListener 类实现了 OnTouchListener 接口,用于处理添加新笔记按钮的触摸事件。
|
|
|
|
|
private class NewNoteOnTouchListener implements OnTouchListener {
|
|
|
|
|
|
|
|
|
|
// onTouch 方法是触摸事件监听器的主要方法,用于处理触摸事件。
|
|
|
|
|
public boolean onTouch(View v, MotionEvent event) {
|
|
|
|
|
switch (event.getAction()) {
|
|
|
|
|
case MotionEvent.ACTION_DOWN: {
|
|
|
|
|
Display display = getWindowManager().getDefaultDisplay(); // 获取默认显示器。
|
|
|
|
|
int screenHeight = display.getHeight(); // 获取屏幕高度。
|
|
|
|
|
int newNoteViewHeight = mAddNewNote.getHeight(); // 获取添加新笔记按钮的高度。
|
|
|
|
|
int start = screenHeight - newNoteViewHeight; // 计算按钮底部的位置。
|
|
|
|
|
int eventY = start + (int) event.getY(); // 计算触摸事件相对于按钮底部的位置。
|
|
|
|
|
|
|
|
|
|
// 如果当前状态是子文件夹,则需要减去标题栏的高度。
|
|
|
|
|
if (mState == ListEditState.SUB_FOLDER) {
|
|
|
|
|
eventY -= mTitleBar.getHeight();
|
|
|
|
|
start -= mTitleBar.getHeight();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* HACKME:这是一个临时解决方案,用于将触摸事件转发到列表视图。
|
|
|
|
|
* 原因是用户可能点击了新笔记按钮的透明部分,这会导致按钮被意外触发。
|
|
|
|
|
* 通过计算触摸事件与按钮的相对位置,我们可以确定是否应该转发触摸事件。
|
|
|
|
|
*/
|
|
|
|
|
if (event.getY() < (event.getX() * (-0.12) + 94)) {
|
|
|
|
|
View view = mNotesListView.getChildAt(mNotesListView.getChildCount() - 1
|
|
|
|
|
- mNotesListView.getFooterViewsCount()); // 获取列表视图中最后一个可见子视图。
|
|
|
|
|
if (view != null && view.getBottom() > start
|
|
|
|
|
&& (view.getTop() < (start + 94))) {
|
|
|
|
|
mOriginY = (int) event.getY(); // 记录原始触摸事件的Y坐标。
|
|
|
|
|
mDispatchY = eventY; // 设置转发触摸事件的Y坐标。
|
|
|
|
|
event.setLocation(event.getX(), mDispatchY); // 设置新的触摸事件位置。
|
|
|
|
|
mDispatch = true; // 设置事件转发标志位。
|
|
|
|
|
return mNotesListView.dispatchTouchEvent(event); // 转发触摸事件到列表视图。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case MotionEvent.ACTION_MOVE: {
|
|
|
|
|
if (mDispatch) {
|
|
|
|
|
mDispatchY += (int) event.getY() - mOriginY; // 更新转发触摸事件的Y坐标。
|
|
|
|
|
event.setLocation(event.getX(), mDispatchY); // 设置新的触摸事件位置。
|
|
|
|
|
return mNotesListView.dispatchTouchEvent(event); // 转发触摸事件到列表视图。
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default: {
|
|
|
|
|
if (mDispatch) {
|
|
|
|
|
event.setLocation(event.getX(), mDispatchY); // 设置触摸事件的新位置。
|
|
|
|
|
mDispatch = false; // 重置事件转发标志位。
|
|
|
|
|
return mNotesListView.dispatchTouchEvent(event); // 转发触摸事件到列表视图。
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// startAsyncNotesListQuery 方法用于启动异步查询笔记列表的操作。
|
|
|
|
|
private void startAsyncNotesListQuery() {
|
|
|
|
|
String selection = (mCurrentFolderId == Notes.ID_ROOT_FOLDER) ? ROOT_FOLDER_SELECTION
|
|
|
|
|
: NORMAL_SELECTION; // 根据当前文件夹 ID 构建查询选择器。
|
|
|
|
|
mBackgroundQueryHandler.startQuery(FOLDER_NOTE_LIST_QUERY_TOKEN, null,
|
|
|
|
|
Notes.CONTENT_NOTE_URI, NoteItemData.PROJECTION, selection, new String[] {
|
|
|
|
|
String.valueOf(mCurrentFolderId)
|
|
|
|
|
}, NoteColumns.TYPE + " DESC," + NoteColumns.MODIFIED_DATE + " DESC");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// BackgroundQueryHandler 类是一个扩展自 AsyncQueryHandler 的类,用于异步处理查询操作。
|
|
|
|
|
private final class BackgroundQueryHandler extends AsyncQueryHandler {
|
|
|
|
|
public BackgroundQueryHandler(ContentResolver contentResolver) {
|
|
|
|
|
super(contentResolver); // 调用父类的构造方法。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onQueryComplete 方法在查询完成时调用,用于处理查询结果。
|
|
|
|
|
@Override
|
|
|
|
|
protected void onQueryComplete(int token, Object cookie, Cursor cursor) {
|
|
|
|
|
switch (token) {
|
|
|
|
|
case FOLDER_NOTE_LIST_QUERY_TOKEN:
|
|
|
|
|
mNotesListAdapter.changeCursor(cursor); // 刷新笔记列表适配器。
|
|
|
|
|
break;
|
|
|
|
|
case FOLDER_LIST_QUERY_TOKEN:
|
|
|
|
|
if (cursor != null && cursor.getCount() > 0) {
|
|
|
|
|
showFolderListMenu(cursor); // 显示文件夹列表菜单。
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "Query folder failed"); // 记录查询文件夹失败的日志。
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// showFolderListMenu 方法用于显示一个对话框,让用户选择要将选中的笔记移动到的文件夹。
|
|
|
|
|
private void showFolderListMenu(Cursor cursor) {
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
builder.setTitle(R.string.menu_title_select_folder);
|
|
|
|
|
final FoldersListAdapter adapter = new FoldersListAdapter(this, cursor);
|
|
|
|
|
builder.setAdapter(adapter, new DialogInterface.OnClickListener() {
|
|
|
|
|
|
|
|
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
|
|
|
DataUtils.batchMoveToFolder(mContentResolver,
|
|
|
|
|
mNotesListAdapter.getSelectedItemIds(), adapter.getItemId(which));
|
|
|
|
|
Toast.makeText(
|
|
|
|
|
NotesListActivity.this,
|
|
|
|
|
getString(R.string.format_move_notes_to_folder,
|
|
|
|
|
mNotesListAdapter.getSelectedCount(),
|
|
|
|
|
adapter.getFolderName(NotesListActivity.this, which)),
|
|
|
|
|
Toast.LENGTH_SHORT).show();
|
|
|
|
|
mModeCallBack.finishActionMode(); // 结束多选模式。
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
builder.show();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// createNewNote 方法用于创建一个新的笔记。
|
|
|
|
|
private void createNewNote() {
|
|
|
|
|
Intent intent = new Intent(this, NoteEditActivity.class);
|
|
|
|
|
intent.setAction(Intent.ACTION_INSERT_OR_EDIT);
|
|
|
|
|
intent.putExtra(Notes.INTENT_EXTRA_FOLDER_ID, mCurrentFolderId);
|
|
|
|
|
this.startActivityForResult(intent, REQUEST_CODE_NEW_NODE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// batchDelete 方法用于批量删除选中的笔记。
|
|
|
|
|
private void batchDelete() {
|
|
|
|
|
new AsyncTask<Void, Void, HashSet<AppWidgetAttribute>>() {
|
|
|
|
|
protected HashSet<AppWidgetAttribute> doInBackground(Void... unused) {
|
|
|
|
|
HashSet<AppWidgetAttribute> widgets = mNotesListAdapter.getSelectedWidget();
|
|
|
|
|
if (!isSyncMode()) {
|
|
|
|
|
// 如果没有同步,直接删除笔记
|
|
|
|
|
if (DataUtils.batchDeleteNotes(mContentResolver, mNotesListAdapter
|
|
|
|
|
.getSelectedItemIds())) {
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "Delete notes error, should not happens"); // 记录删除笔记失败的日志。
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// 如果处于同步模式,将删除的笔记移动到回收站文件夹
|
|
|
|
|
if (!DataUtils.batchMoveToFolder(mContentResolver, mNotesListAdapter
|
|
|
|
|
.getSelectedItemIds(), Notes.ID_TRASH_FOLER)) {
|
|
|
|
|
Log.e(TAG, "Move notes to trash folder error, should not happens"); // 记录移动笔记到回收站失败的日志。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return widgets;
|
|
|
|
|
}
|
|
|
|
|
}.execute();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
// onPostExecute 方法在异步任务完成后调用,用于处理批量删除笔记后的更新小部件操作。
|
|
|
|
|
protected void onPostExecute(HashSet<AppWidgetAttribute> widgets) {
|
|
|
|
|
if (widgets != null) {
|
|
|
|
|
for (AppWidgetAttribute widget : widgets) {
|
|
|
|
|
if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID
|
|
|
|
|
&& widget.widgetType != Notes.TYPE_WIDGET_INVALIDE) {
|
|
|
|
|
updateWidget(widget.widgetId, widget.widgetType); // 更新小部件的数据。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
mModeCallBack.finishActionMode(); // 结束多选模式。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// deleteFolder 方法用于删除指定的文件夹。
|
|
|
|
|
private void deleteFolder(long folderId) {
|
|
|
|
|
if (folderId == Notes.ID_ROOT_FOLDER) {
|
|
|
|
|
Log.e(TAG, "Wrong folder id, should not happen " + folderId); // 如果尝试删除根文件夹,记录错误日志。
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HashSet<Long> ids = new HashSet<Long>();
|
|
|
|
|
ids.add(folderId); // 将文件夹 ID 添加到 ID 集合中。
|
|
|
|
|
HashSet<AppWidgetAttribute> widgets = DataUtils.getFolderNoteWidget(mContentResolver,
|
|
|
|
|
folderId); // 获取文件夹相关的小部件属性。
|
|
|
|
|
if (!isSyncMode()) {
|
|
|
|
|
// 如果没有同步,直接删除文件夹
|
|
|
|
|
DataUtils.batchDeleteNotes(mContentResolver, ids);
|
|
|
|
|
} else {
|
|
|
|
|
// 如果在同步模式下,将删除的文件夹移动到回收站文件夹
|
|
|
|
|
DataUtils.batchMoveToFolder(mContentResolver, ids, Notes.ID_TRASH_FOLER);
|
|
|
|
|
}
|
|
|
|
|
if (widgets != null) {
|
|
|
|
|
for (AppWidgetAttribute widget : widgets) {
|
|
|
|
|
if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID
|
|
|
|
|
&& widget.widgetType != Notes.TYPE_WIDGET_INVALIDE) {
|
|
|
|
|
updateWidget(widget.widgetId, widget.widgetType); // 更新小部件的数据。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// openNode 方法用于打开一个笔记。
|
|
|
|
|
private void openNode(NoteItemData data) {
|
|
|
|
|
Intent intent = new Intent(this, NoteEditActivity.class);
|
|
|
|
|
intent.setAction(Intent.ACTION_VIEW);
|
|
|
|
|
intent.putExtra(Intent.EXTRA_UID, data.getId()); // 传递笔记的 ID 作为 Intent 的一部分。
|
|
|
|
|
this.startActivityForResult(intent, REQUEST_CODE_OPEN_NODE); // 启动活动并返回结果。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// openFolder 方法用于打开一个文件夹。
|
|
|
|
|
private void openFolder(NoteItemData data) {
|
|
|
|
|
mCurrentFolderId = data.getId(); // 更新当前文件夹 ID。
|
|
|
|
|
startAsyncNotesListQuery(); // 启动异步查询以获取文件夹中的笔记列表。
|
|
|
|
|
if (data.getId() == Notes.ID_CALL_RECORD_FOLDER) {
|
|
|
|
|
mState = ListEditState.CALL_RECORD_FOLDER; // 设置状态为通话记录文件夹。
|
|
|
|
|
mAddNewNote.setVisibility(View.GONE); // 隐藏添加新笔记按钮。
|
|
|
|
|
} else {
|
|
|
|
|
mState = ListEditState.SUB_FOLDER; // 设置状态为子文件夹。
|
|
|
|
|
}
|
|
|
|
|
if (data.getId() == Notes.ID_CALL_RECORD_FOLDER) {
|
|
|
|
|
mTitleBar.setText(R.string.call_record_folder_name); // 设置标题栏文本为通话记录文件夹名称。
|
|
|
|
|
} else {
|
|
|
|
|
mTitleBar.setText(data.getSnippet()); // 设置标题栏文本为文件夹的片段。
|
|
|
|
|
}
|
|
|
|
|
mTitleBar.setVisibility(View.VISIBLE); // 显示标题栏。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onClick 方法是 View.OnClickListener 接口的方法,用于处理按钮点击事件。
|
|
|
|
|
public void onClick(View v) {
|
|
|
|
|
switch (v.getId()) {
|
|
|
|
|
case R.id.btn_new_note:
|
|
|
|
|
createNewNote(); // 当点击添加新笔记按钮时,创建新笔记。
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break; // 处理其他按钮点击事件。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// showSoftInput 方法用于显示软键盘。
|
|
|
|
|
private void showSoftInput() {
|
|
|
|
|
InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
|
|
|
|
|
if (inputMethodManager != null) {
|
|
|
|
|
inputMethodManager.toggleSoftInput(InputMethodManager.SHOW_FORCED, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// hideSoftInput 方法用于隐藏软键盘。
|
|
|
|
|
private void hideSoftInput(View view) {
|
|
|
|
|
InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
|
|
|
|
|
if (inputMethodManager != null) {
|
|
|
|
|
inputMethodManager.hideSoftInputFromWindow(view.getWindowToken(), 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// showCreateOrModifyFolderDialog 方法用于显示创建或修改文件夹的对话框。
|
|
|
|
|
private void showCreateOrModifyFolderDialog(final boolean create) {
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
|
|
|
|
View view = LayoutInflater.from(this).inflate(R.layout.dialog_edit_text, null);
|
|
|
|
|
final EditText etName = (EditText) view.findViewById(R.id.et_foler_name);
|
|
|
|
|
|
|
|
|
|
showSoftInput(); // 显示软键盘。
|
|
|
|
|
|
|
|
|
|
if (!create) {
|
|
|
|
|
if (mFocusNoteDataItem != null) {
|
|
|
|
|
etName.setText(mFocusNoteDataItem.getSnippet()); // 如果不是创建文件夹,设置输入框为当前焦点项的片段。
|
|
|
|
|
builder.setTitle(getString(R.string.menu_folder_change_name));
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "The long click data item is null"); // 记录错误日志。
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
etName.setText(""); // 如果是创建文件夹,清空输入框。
|
|
|
|
|
builder.setTitle(this.getString(R.string.menu_create_folder));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok, null);
|
|
|
|
|
builder.setNegativeButton(android.R.string.cancel, new DialogInterface.OnClickListener() {
|
|
|
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
|
|
|
hideSoftInput(etName); // 当用户点击取消按钮时,隐藏软键盘。
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
final Dialog dialog = builder.setView(view).show(); // 显示对话框。
|
|
|
|
|
Button positive = (Button)dialog.findViewById(android.R.id.button1);
|
|
|
|
|
positive.setOnClickListener(new OnClickListener() {
|
|
|
|
|
public void onClick(View v) {
|
|
|
|
|
hideSoftInput(etName); // 当用户点击确定按钮时,隐藏软键盘。
|
|
|
|
|
String name = etName.getText().toString();
|
|
|
|
|
if (DataUtils.checkVisibleFolderName(mContentResolver, name)) {
|
|
|
|
|
Toast.makeText(NotesListActivity.this, getString(R.string.folder_exist, name),
|
|
|
|
|
Toast.LENGTH_LONG).show(); // 如果文件夹名称已存在,显示提示信息。
|
|
|
|
|
etName.setSelection(0, etName.length()); // 选择输入框中的文本。
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!create) {
|
|
|
|
|
if (!TextUtils.isEmpty(name)) {
|
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
|
values.put(NoteColumns.SNIPPET, name);
|
|
|
|
|
values.put(NoteColumns.TYPE, Notes.TYPE_FOLDER);
|
|
|
|
|
values.put(NoteColumns.LOCAL_MODIFIED, 1);
|
|
|
|
|
mContentResolver.update(Notes.CONTENT_NOTE_URI, values, NoteColumns.ID
|
|
|
|
|
+ "=?", new String[] {
|
|
|
|
|
String.valueOf(mFocusNoteDataItem.getId())
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else if (!TextUtils.isEmpty(name)) {
|
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
|
values.put(NoteColumns.SNIPPET, name);
|
|
|
|
|
values.put(NoteColumns.TYPE, Notes.TYPE_FOLDER);
|
|
|
|
|
mContentResolver.insert(Notes.CONTENT_NOTE_URI, values); // 插入新的文件夹。
|
|
|
|
|
}
|
|
|
|
|
dialog.dismiss(); // 关闭对话框。
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (TextUtils.isEmpty(etName.getText())) {
|
|
|
|
|
positive.setEnabled(false); // 如果输入框为空,禁用确定按钮。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 当输入框的内容变化时,更新确定按钮的可用性。
|
|
|
|
|
*/
|
|
|
|
|
etName.addTextChangedListener(new TextWatcher() {
|
|
|
|
|
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
|
|
|
|
|
// TODO Auto-generated method stub
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void onTextChanged(CharSequence s, int start, int before, int count) {
|
|
|
|
|
if (TextUtils.isEmpty(etName.getText())) {
|
|
|
|
|
positive.setEnabled(false); // 如果输入框为空,禁用确定按钮。
|
|
|
|
|
} else {
|
|
|
|
|
positive.setEnabled(true); // 否则,启用确定按钮。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void afterTextChanged(Editable s) {
|
|
|
|
|
// TODO Auto-generated method stub
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
// onBackPressed 方法是 Activity 生命周期中的一个回调方法,当用户按下返回按钮时调用。
|
|
|
|
|
@Override
|
|
|
|
|
public void onBackPressed() {
|
|
|
|
|
switch (mState) { // 根据当前状态处理返回按钮。
|
|
|
|
|
case SUB_FOLDER: // 如果当前状态是子文件夹。
|
|
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER; // 更新当前文件夹 ID 为根文件夹。
|
|
|
|
|
mState = ListEditState.NOTE_LIST; // 更新状态为笔记列表。
|
|
|
|
|
startAsyncNotesListQuery(); // 启动异步查询以获取笔记列表。
|
|
|
|
|
mTitleBar.setVisibility(View.GONE); // 隐藏标题栏。
|
|
|
|
|
break;
|
|
|
|
|
case CALL_RECORD_FOLDER: // 如果当前状态是通话记录文件夹。
|
|
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER; // 更新当前文件夹 ID 为根文件夹。
|
|
|
|
|
mState = ListEditState.NOTE_LIST; // 更新状态为笔记列表。
|
|
|
|
|
mAddNewNote.setVisibility(View.VISIBLE); // 显示添加新笔记按钮。
|
|
|
|
|
mTitleBar.setVisibility(View.GONE); // 隐藏标题栏。
|
|
|
|
|
startAsyncNotesListQuery(); // 启动异步查询以获取笔记列表。
|
|
|
|
|
break;
|
|
|
|
|
case NOTE_LIST: // 如果当前状态是笔记列表。
|
|
|
|
|
super.onBackPressed(); // 调用父类的 onBackPressed 方法。
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// updateWidget 方法用于更新小部件的数据。
|
|
|
|
|
private void updateWidget(int appWidgetId, int appWidgetType) {
|
|
|
|
|
Intent intent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
|
|
|
|
|
if (appWidgetType == Notes.TYPE_WIDGET_2X) {
|
|
|
|
|
intent.setClass(this, NoteWidgetProvider_2x.class); // 设置意图的类为 2x 类型的小部件提供者。
|
|
|
|
|
} else if (appWidgetType == Notes.TYPE_WIDGET_4X) {
|
|
|
|
|
intent.setClass(this, NoteWidgetProvider_4x.class); // 设置意图的类为 4x 类型的小部件提供者。
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "Unspported widget type"); // 记录不支持的类型日志。
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, new int[] {
|
|
|
|
|
appWidgetId
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
sendBroadcast(intent); // 发送广播以更新小部件。
|
|
|
|
|
setResult(RESULT_OK, intent); // 设置结果为成功。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// OnCreateContextMenuListener 接口的实现,用于为文件夹上下文菜单创建项。
|
|
|
|
|
private final OnCreateContextMenuListener mFolderOnCreateContextMenuListener = new OnCreateContextMenuListener() {
|
|
|
|
|
public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) {
|
|
|
|
|
if (mFocusNoteDataItem != null) { // 如果当前焦点项不为空。
|
|
|
|
|
menu.setHeaderTitle(mFocusNoteDataItem.getSnippet()); // 设置上下文菜单的标题为焦点项的片段。
|
|
|
|
|
menu.add(0, MENU_FOLDER_VIEW, 0, R.string.menu_folder_view); // 添加查看文件夹菜单项。
|
|
|
|
|
menu.add(0, MENU_FOLDER_DELETE, 0, R.string.menu_folder_delete); // 添加删除文件夹菜单项。
|
|
|
|
|
menu.add(0, MENU_FOLDER_CHANGE_NAME, 0, R.string.menu_folder_change_name); // 添加修改文件夹名称菜单项。
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// onContextMenuClosed 方法在上下文菜单关闭时调用,用于清理状态。
|
|
|
|
|
@Override
|
|
|
|
|
public void onContextMenuClosed(Menu menu) {
|
|
|
|
|
if (mNotesListView != null) { // 如果列表视图不为空。
|
|
|
|
|
mNotesListView.setOnCreateContextMenuListener(null); // 清除列表视图的上下文菜单监听器。
|
|
|
|
|
}
|
|
|
|
|
super.onContextMenuClosed(menu); // 调用父类的 onContextMenuClosed 方法。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onContextItemSelected 方法在用户选择上下文菜单项时调用,用于处理文件夹上下文菜单项的点击事件。
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onContextItemSelected(MenuItem item) {
|
|
|
|
|
if (mFocusNoteDataItem == null) { // 如果当前焦点项为空。
|
|
|
|
|
Log.e(TAG, "The long click data item is null"); // 记录错误日志。
|
|
|
|
|
return false; // 返回 false,表示不处理该事件。
|
|
|
|
|
}
|
|
|
|
|
switch (item.getItemId()) { // 处理上下文菜单项的点击事件。
|
|
|
|
|
case MENU_FOLDER_VIEW:
|
|
|
|
|
openFolder(mFocusNoteDataItem); // 打开文件夹。
|
|
|
|
|
break;
|
|
|
|
|
case MENU_FOLDER_DELETE:
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
|
|
|
|
builder.setTitle(getString(R.string.alert_title_delete));
|
|
|
|
|
builder.setIcon(android.R.drawable.ic_dialog_alert);
|
|
|
|
|
builder.setMessage(getString(R.string.alert_message_delete_folder));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok,
|
|
|
|
|
new DialogInterface.OnClickListener() {
|
|
|
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
|
|
|
deleteFolder(mFocusNoteDataItem.getId()); // 删除文件夹。
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
builder.setNegativeButton(android.R.string.cancel, null);
|
|
|
|
|
builder.show(); // 显示对话框。
|
|
|
|
|
break;
|
|
|
|
|
case MENU_FOLDER_CHANGE_NAME:
|
|
|
|
|
showCreateOrModifyFolderDialog(false); // 显示修改文件夹名称的对话框。
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true; // 返回 true,表示处理了该事件。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onPrepareOptionsMenu 方法用于准备选项菜单,根据当前状态填充不同的菜单项。
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onPrepareOptionsMenu(Menu menu) {
|
|
|
|
|
menu.clear(); // 清空当前菜单项。
|
|
|
|
|
if (mState == ListEditState.NOTE_LIST) {
|
|
|
|
|
getMenuInflater().inflate(R.menu.note_list, menu); // 加载笔记列表的菜单项。
|
|
|
|
|
// 设置同步或取消同步的选项
|
|
|
|
|
menu.findItem(R.id.menu_sync).setTitle(
|
|
|
|
|
GTaskSyncService.isSyncing() ? R.string.menu_sync_cancel : R.string.menu_sync);
|
|
|
|
|
} else if (mState == ListEditState.SUB_FOLDER) {
|
|
|
|
|
getMenuInflater().inflate(R.menu.sub_folder, menu); // 加载子文件夹的菜单项。
|
|
|
|
|
} else if (mState == ListEditState.CALL_RECORD_FOLDER) {
|
|
|
|
|
getMenuInflater().inflate(R.menu.call_record_folder, menu); // 加载通话记录文件夹的菜单项。
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "Wrong state:" + mState); // 如果状态错误,记录日志。
|
|
|
|
|
}
|
|
|
|
|
return true; // 返回 true,表示处理了该事件。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onOptionsItemSelected 方法用于处理选项菜单项的点击事件。
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onOptionsItemSelected(MenuItem item) {
|
|
|
|
|
switch (item.getItemId()) {
|
|
|
|
|
case R.id.menu_new_folder:
|
|
|
|
|
showCreateOrModifyFolderDialog(true); // 显示创建文件夹的对话框。
|
|
|
|
|
break;
|
|
|
|
|
case R.id.menu_export_text:
|
|
|
|
|
exportNoteToText(); // 导出笔记到文本文件。
|
|
|
|
|
break;
|
|
|
|
|
case R.id.menu_sync:
|
|
|
|
|
if (isSyncMode()) {
|
|
|
|
|
if (TextUtils.equals(item.getTitle(), getString(R.string.menu_sync))) {
|
|
|
|
|
GTaskSyncService.startSync(this); // 启动同步服务。
|
|
|
|
|
} else {
|
|
|
|
|
GTaskSyncService.cancelSync(this); // 取消同步服务。
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
startPreferenceActivity(); // 启动首选项活动。
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case R.id.menu_setting:
|
|
|
|
|
startPreferenceActivity(); // 启动首选项活动。
|
|
|
|
|
break;
|
|
|
|
|
case R.id.menu_new_note:
|
|
|
|
|
createNewNote(); // 创建新笔记。
|
|
|
|
|
break;
|
|
|
|
|
case R.id.menu_search:
|
|
|
|
|
onSearchRequested(); // 请求搜索。
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return true; // 返回 true,表示处理了该事件。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onSearchRequested 方法是 Activity 生命周期中的一个回调方法,当用户请求搜索时调用。
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onSearchRequested() {
|
|
|
|
|
startSearch(null, false, null /* appData */, false); // 启动搜索活动。
|
|
|
|
|
return true; // 返回 true,表示处理了该事件。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// exportNoteToText 方法用于导出笔记到文本文件。
|
|
|
|
|
private void exportNoteToText() {
|
|
|
|
|
final BackupUtils backup = BackupUtils.getInstance(NotesListActivity.this);
|
|
|
|
|
new AsyncTask<Void, Void, Integer>() {
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected Integer doInBackground(Void... unused) {
|
|
|
|
|
return backup.exportToText(); // 异步导出笔记到文本文件。
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onPostExecute(Integer result) {
|
|
|
|
|
if (result == BackupUtils.STATE_SD_CARD_UNMOUONTED) {
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
builder.setTitle(getString(R.string.failed_sdcard_export));
|
|
|
|
|
builder.setMessage(getString(R.string.error_sdcard_unmounted));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok, null);
|
|
|
|
|
builder.show();
|
|
|
|
|
} else if (result == BackupUtils.STATE_SUCCESS) {
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
builder.setTitle(getString(R.string.success_sdcard_export));
|
|
|
|
|
builder.setMessage(getString(
|
|
|
|
|
R.string.format_exported_file_location, backup.getExportedTextFileName(), backup
|
|
|
|
|
.getExportedTextFileDir()));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok, null);
|
|
|
|
|
builder.show();
|
|
|
|
|
} else if (result == BackupUtils.STATE_SYSTEM_ERROR) {
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
builder.setTitle(getString(R.string.failed_sdcard_export));
|
|
|
|
|
builder.setMessage(getString(R.string.error_sdcard_export));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok, null);
|
|
|
|
|
builder.show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}.execute();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// isSyncMode 方法用于检查是否处于同步模式。
|
|
|
|
|
private boolean isSyncMode() {
|
|
|
|
|
return NotesPreferenceActivity.getSyncAccountName(this).trim().length() > 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// startPreferenceActivity 方法用于启动首选项活动。
|
|
|
|
|
private void startPreferenceActivity() {
|
|
|
|
|
Activity from = getParent() != null ? getParent() : this;
|
|
|
|
|
Intent intent = new Intent(from, NotesPreferenceActivity.class);
|
|
|
|
|
from.startActivityIfNeeded(intent, -1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// OnListItemClickListener 类实现了 OnItemClickListener 接口,用于处理列表项的点击事件。
|
|
|
|
|
private class OnListItemClickListener implements OnItemClickListener {
|
|
|
|
|
|
|
|
|
|
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
|
|
|
|
|
if (view instanceof NotesListItem) {
|
|
|
|
|
NoteItemData item = ((NotesListItem) view).getItemData();
|
|
|
|
|
if (mNotesListAdapter.isInChoiceMode()) {
|
|
|
|
|
if (item.getType() == Notes.TYPE_NOTE) {
|
|
|
|
|
position = position - mNotesListView.getHeaderViewsCount();
|
|
|
|
|
mModeCallBack.onItemCheckedStateChanged(null, position, id,
|
|
|
|
|
!mNotesListAdapter.isSelectedItem(position));
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (mState) {
|
|
|
|
|
case NOTE_LIST:
|
|
|
|
|
if (item.getType() == Notes.TYPE_FOLDER
|
|
|
|
|
|| item.getType() == Notes.TYPE_SYSTEM) {
|
|
|
|
|
openFolder(item);
|
|
|
|
|
} else if (item.getType() == Notes.TYPE_NOTE) {
|
|
|
|
|
openNode(item);
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "Wrong note type in NOTE_LIST");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case SUB_FOLDER:
|
|
|
|
|
case CALL_RECORD_FOLDER:
|
|
|
|
|
if (item.getType() == Notes.TYPE_NOTE) {
|
|
|
|
|
openNode(item);
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "Wrong note type in SUB_FOLDER");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// startQueryDestinationFolders 方法用于启动查询目的地文件夹的异步查询。
|
|
|
|
|
private void startQueryDestinationFolders() {
|
|
|
|
|
String selection = NoteColumns.TYPE + "=? AND " + NoteColumns.PARENT_ID + "<>? AND " + NoteColumns.ID + "<>?";
|
|
|
|
|
selection = (mState == ListEditState.NOTE_LIST) ? selection:
|
|
|
|
|
"(" + selection + ") OR (" + NoteColumns.ID + "=" + Notes.ID_ROOT_FOLDER + ")";
|
|
|
|
|
|
|
|
|
|
mBackgroundQueryHandler.startQuery(FOLDER_LIST_QUERY_TOKEN,
|
|
|
|
|
null,
|
|
|
|
|
Notes.CONTENT_NOTE_URI,
|
|
|
|
|
FoldersListAdapter.PROJECTION,
|
|
|
|
|
selection,
|
|
|
|
|
new String[] {
|
|
|
|
|
String.valueOf(Notes.TYPE_FOLDER),
|
|
|
|
|
String.valueOf(Notes.ID_TRASH_FOLER),
|
|
|
|
|
String.valueOf(mCurrentFolderId)
|
|
|
|
|
},
|
|
|
|
|
NoteColumns.MODIFIED_DATE + " DESC");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// onItemLongClick 方法用于处理列表项的长按点击事件。
|
|
|
|
|
public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) {
|
|
|
|
|
if (view instanceof NotesListItem) {
|
|
|
|
|
mFocusNoteDataItem = ((NotesListItem) view).getItemData();
|
|
|
|
|
if (mFocusNoteDataItem.getType() == Notes.TYPE_NOTE && !mNotesListAdapter.isInChoiceMode()) {
|
|
|
|
|
if (mNotesListView.startActionMode(mModeCallBack) != null) {
|
|
|
|
|
mModeCallBack.onItemCheckedStateChanged(null, position, id, true);
|
|
|
|
|
mNotesListView.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
|
|
|
|
|
} else {
|
|
|
|
|
Log.e(TAG, "startActionMode fails");
|
|
|
|
|
}
|
|
|
|
|
} else if (mFocusNoteDataItem.getType() == Notes.TYPE_FOLDER) {
|
|
|
|
|
mNotesListView.setOnCreateContextMenuListener(mFolderOnCreateContextMenuListener);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|