|
|
/*
|
|
|
* 版权所有 (c) 2010-2011,MiCode 开源社区 (www.micode.net)
|
|
|
* 根据 Apache 许可证 2.0 版本("许可证")授权;
|
|
|
* 除非符合许可证的规定,否则不得使用本文件。
|
|
|
* 您可以从以下网址获取许可证副本:
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
* 除非适用法律要求或书面同意,本软件按"原样"分发,
|
|
|
* 没有任何明示或暗示的保证或条件。
|
|
|
* 详见许可证中规定的权限和限制。
|
|
|
* (注:这是一份标准的Apache许可证2.0版本的开源声明)
|
|
|
*/
|
|
|
|
|
|
// 定义当前包名为net.micode.notes.ui
|
|
|
package net.micode.notes.ui;
|
|
|
|
|
|
/* 导入Android基础包 */
|
|
|
import android.annotation.SuppressLint;
|
|
|
import android.app.Activity; // 基础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.Intent; // Intent类
|
|
|
import android.content.SharedPreferences; // 共享偏好设置
|
|
|
import android.database.Cursor; // 数据库游标
|
|
|
import android.os.AsyncTask; // 异步任务类
|
|
|
import android.os.Bundle; // 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.TextView; // 文本视图
|
|
|
import android.widget.Toast; // 提示信息控件
|
|
|
|
|
|
/* 导入AndroidX支持库 */
|
|
|
import androidx.annotation.NonNull; // 非空注解
|
|
|
|
|
|
/* 导入项目资源 */
|
|
|
import net.micode.notes.R; // 自动生成的R资源类
|
|
|
|
|
|
/* 导入项目数据相关类 */
|
|
|
import net.micode.notes.data.Notes; // 笔记数据库常量
|
|
|
import net.micode.notes.data.Notes.NoteColumns; // 笔记列名定义
|
|
|
|
|
|
/* 导入Google任务同步服务 */
|
|
|
import net.micode.notes.gtask.remote.GTaskSyncService; // Google任务同步服务
|
|
|
|
|
|
/* 导入项目模型类 */
|
|
|
import net.micode.notes.model.WorkingNote; // 工作笔记模型
|
|
|
|
|
|
/* 导入项目工具类 */
|
|
|
import net.micode.notes.tool.BackupUtils; // 备份工具
|
|
|
import net.micode.notes.tool.DataUtils; // 数据工具
|
|
|
import net.micode.notes.tool.ResourceParser; // 资源解析器
|
|
|
|
|
|
/* 导入项目UI适配器 */
|
|
|
import net.micode.notes.ui.NotesListAdapter.AppWidgetAttribute; // 小部件属性定义
|
|
|
|
|
|
/* 导入项目小部件相关 */
|
|
|
import net.micode.notes.widget.NoteWidgetProvider_2x; // 2x尺寸小部件
|
|
|
import net.micode.notes.widget.NoteWidgetProvider_4x; // 4x尺寸小部件
|
|
|
|
|
|
/* 导入Java IO类 */
|
|
|
import java.io.BufferedReader; // 缓冲读取器
|
|
|
import java.io.IOException; // IO异常
|
|
|
import java.io.InputStream; // 输入流
|
|
|
import java.io.InputStreamReader; // 输入流读取器
|
|
|
|
|
|
/* 导入Java集合类 */
|
|
|
import java.util.HashSet; // 哈希集合
|
|
|
|
|
|
// 定义笔记列表Activity,实现点击和长按监听接口
|
|
|
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; // 文件夹列表查询标识
|
|
|
|
|
|
// 上下文菜单项ID
|
|
|
private static final int MENU_FOLDER_DELETE = 0; // 删除文件夹菜单ID
|
|
|
private static final int MENU_FOLDER_VIEW = 1; // 查看文件夹菜单ID
|
|
|
private static final int MENU_FOLDER_CHANGE_NAME = 2; // 重命名文件夹菜单ID
|
|
|
|
|
|
// 偏好设置键(是否已添加引导说明)
|
|
|
private static final String PREFERENCE_ADD_INTRODUCTION = "net.micode.notes.introduction";
|
|
|
|
|
|
// 列表编辑状态枚举
|
|
|
private enum ListEditState {
|
|
|
NOTE_LIST, // 普通笔记列表模式
|
|
|
SUB_FOLDER, // 子文件夹模式
|
|
|
CALL_RECORD_FOLDER // 通话记录文件夹模式
|
|
|
}
|
|
|
|
|
|
// 成员变量声明
|
|
|
private ListEditState mState; // 当前列表状态
|
|
|
private BackgroundQueryHandler mBackgroundQueryHandler; // 后台查询处理器
|
|
|
private NotesListAdapter mNotesListAdapter; // 笔记列表适配器
|
|
|
private ListView mNotesListView; // 笔记列表视图
|
|
|
private Button mAddNewNote; // 新建笔记按钮
|
|
|
private boolean mDispatch; // 触摸事件分发标志
|
|
|
private int mOriginY; // 触摸起始Y坐标
|
|
|
private int mDispatchY; // 触摸分发Y坐标
|
|
|
private TextView mTitleBar; // 标题栏文本
|
|
|
private long mCurrentFolderId; // 当前文件夹ID
|
|
|
private ContentResolver mContentResolver; // 内容解析器
|
|
|
private ModeCallback mModeCallBack; // 多选模式回调
|
|
|
private static final String TAG = "NotesListActivity"; // 日志标签
|
|
|
public static final int NOTES_LISTVIEW_SCROLL_RATE = 30; // 列表滚动速率
|
|
|
private NoteItemData mFocusNoteDataItem; // 当前焦点笔记数据
|
|
|
|
|
|
// SQL查询条件
|
|
|
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; // 新建笔记请求码
|
|
|
|
|
|
@Override
|
|
|
protected void onCreate(Bundle savedInstanceState) {
|
|
|
super.onCreate(savedInstanceState);
|
|
|
setContentView(R.layout.note_list); // 设置布局文件
|
|
|
|
|
|
initResources(); // 初始化资源
|
|
|
|
|
|
// 首次使用时添加引导说明
|
|
|
setAppInfoFromRawRes();
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
|
// 处理笔记打开/新建返回结果
|
|
|
if (resultCode == RESULT_OK
|
|
|
&& (requestCode == REQUEST_CODE_OPEN_NODE || requestCode == REQUEST_CODE_NEW_NODE)) {
|
|
|
mNotesListAdapter.changeCursor(null); // 清空当前游标触发刷新
|
|
|
} else {
|
|
|
super.onActivityResult(requestCode, resultCode, data);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 从raw资源读取并设置应用引导信息
|
|
|
private void setAppInfoFromRawRes() {
|
|
|
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this);
|
|
|
// 检查是否已经添加过引导说明
|
|
|
if (!sp.getBoolean(PREFERENCE_ADD_INTRODUCTION, false)) {
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
InputStream in = null;
|
|
|
try {
|
|
|
// 从raw资源读取引导文本
|
|
|
in = getResources().openRawResource(R.raw.introduction);
|
|
|
InputStreamReader isr = new InputStreamReader(in);
|
|
|
BufferedReader br = new BufferedReader(isr);
|
|
|
char[] buf = new char[1024];
|
|
|
int len;
|
|
|
while ((len = br.read(buf)) > 0) {
|
|
|
sb.append(buf, 0, len);
|
|
|
}
|
|
|
} catch (IOException e) {
|
|
|
e.printStackTrace();
|
|
|
return;
|
|
|
} finally {
|
|
|
// 关闭输入流
|
|
|
if (in != null) {
|
|
|
try {
|
|
|
in.close();
|
|
|
} catch (IOException e) {
|
|
|
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).apply();
|
|
|
} else {
|
|
|
Log.e(TAG, "Save introduction note error");
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
protected void onStart() {
|
|
|
super.onStart();
|
|
|
startAsyncNotesListQuery(); // 启动异步笔记列表查询
|
|
|
}
|
|
|
|
|
|
// 初始化视图资源和变量
|
|
|
@SuppressLint("ClickableViewAccessibility")
|
|
|
private void initResources() {
|
|
|
mContentResolver = this.getContentResolver();
|
|
|
mBackgroundQueryHandler = new BackgroundQueryHandler(this.getContentResolver());
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER; // 默认设为根文件夹
|
|
|
|
|
|
// 初始化列表视图
|
|
|
mNotesListView = 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);
|
|
|
mNotesListView.setAdapter(mNotesListAdapter);
|
|
|
|
|
|
// 初始化新建笔记按钮
|
|
|
mAddNewNote = findViewById(R.id.btn_new_note);
|
|
|
mAddNewNote.setOnClickListener(this);
|
|
|
mAddNewNote.setOnTouchListener(new NewNoteOnTouchListener());
|
|
|
|
|
|
// 初始化触摸相关变量
|
|
|
mDispatch = false;
|
|
|
mDispatchY = 0;
|
|
|
mOriginY = 0;
|
|
|
|
|
|
// 初始化标题栏
|
|
|
mTitleBar = findViewById(R.id.tv_title_bar);
|
|
|
mState = ListEditState.NOTE_LIST; // 初始状态为普通笔记列表
|
|
|
|
|
|
// 初始化多选模式回调
|
|
|
mModeCallBack = new ModeCallback();
|
|
|
}
|
|
|
|
|
|
// 多选模式回调实现类
|
|
|
private class ModeCallback implements ListView.MultiChoiceModeListener, OnMenuItemClickListener {
|
|
|
private DropdownMenu mDropDownMenu; // 下拉菜单
|
|
|
private ActionMode mActionMode; // 操作模式
|
|
|
|
|
|
@Override
|
|
|
public boolean onCreateActionMode(ActionMode mode, Menu menu) {
|
|
|
// 加载多选操作菜单
|
|
|
getMenuInflater().inflate(R.menu.note_list_options, menu);
|
|
|
menu.findItem(R.id.delete).setOnMenuItemClickListener(this);
|
|
|
|
|
|
// 初始化移动菜单
|
|
|
// 移动菜单项
|
|
|
MenuItem 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;
|
|
|
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);
|
|
|
|
|
|
// 初始化下拉选择菜单
|
|
|
mDropDownMenu = new DropdownMenu(NotesListActivity.this,
|
|
|
customView.findViewById(R.id.selection_menu),
|
|
|
R.menu.note_list_dropdown);
|
|
|
mDropDownMenu.setOnDropdownMenuItemClickListener(item -> {
|
|
|
// 全选/反选处理
|
|
|
mNotesListAdapter.selectAll(!mNotesListAdapter.isAllSelected());
|
|
|
updateMenu();
|
|
|
return true;
|
|
|
});
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 更新多选操作菜单的状态显示
|
|
|
* 1. 刷新选中项数量显示
|
|
|
* 2. 更新全选/反选按钮状态
|
|
|
*/
|
|
|
private void updateMenu() {
|
|
|
// 获取当前选中的笔记项数量(从列表适配器中获取)
|
|
|
int selectedCount = mNotesListAdapter.getSelectedCount();
|
|
|
|
|
|
/**
|
|
|
* 设置下拉菜单标题文本
|
|
|
* 使用字符串资源 R.string.menu_select_title 作为模板
|
|
|
* 将选中数量 selectedCount 作为参数插入字符串
|
|
|
* 示例:当选中3项时显示"已选3项"
|
|
|
*/
|
|
|
String format = getResources().getString(R.string.menu_select_title, selectedCount);
|
|
|
mDropDownMenu.setTitle(format); // 将格式化后的字符串设置为菜单标题
|
|
|
|
|
|
// 从下拉菜单中获取全选功能菜单项(通过资源ID R.id.action_select_all )
|
|
|
MenuItem item = mDropDownMenu.findItem(R.id.action_select_all);
|
|
|
|
|
|
// 确保菜单项存在(防御性编程)
|
|
|
if (item != null) {
|
|
|
/**
|
|
|
* 检查是否所有笔记项都被选中
|
|
|
* 根据状态更新菜单项:
|
|
|
* 1. 已全选:显示选中状态和"取消全选"文本
|
|
|
* 2. 未全选:显示未选中状态和"全选"文本
|
|
|
*/
|
|
|
if (mNotesListAdapter.isAllSelected()) {
|
|
|
item.setChecked(true); // 显示复选框选中状态
|
|
|
item.setTitle(R.string.menu_deselect_all); // 设置文本为"取消全选"
|
|
|
} else {
|
|
|
item.setChecked(false); // 显示复选框未选中状态
|
|
|
item.setTitle(R.string.menu_select_all); // 设置文本为"全选"
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* ActionMode准备时的回调方法
|
|
|
* @param mode 当前的ActionMode对象
|
|
|
* @param menu 要显示的菜单
|
|
|
* @return boolean 返回false表示不需要重新创建菜单
|
|
|
*/
|
|
|
public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
|
|
|
// TODO: 需要在此处添加菜单项的动态更新逻辑
|
|
|
return false; // 默认返回false
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* ActionMode菜单项点击事件处理
|
|
|
* @param mode 当前的ActionMode对象
|
|
|
* @param item 被点击的菜单项
|
|
|
* @return boolean 返回false表示事件未处理,会继续传递
|
|
|
*/
|
|
|
public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
|
|
|
// TODO: 需要添加其他菜单项的点击处理逻辑
|
|
|
return false; // 默认返回false
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 销毁ActionMode时的清理操作
|
|
|
* @param mode 要销毁的ActionMode对象
|
|
|
*/
|
|
|
public void onDestroyActionMode(ActionMode mode) {
|
|
|
mNotesListAdapter.setChoiceMode(false); // 禁用适配器的多选模式
|
|
|
mNotesListView.setLongClickable(true); // 恢复列表视图的长按功能
|
|
|
mAddNewNote.setVisibility(View.VISIBLE); // 显示新建笔记按钮
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 主动结束ActionMode
|
|
|
*/
|
|
|
public void finishActionMode() {
|
|
|
mActionMode.finish(); // 调用当前ActionMode的finish方法结束多选模式
|
|
|
}
|
|
|
/**
|
|
|
* 列表项选中状态变化回调
|
|
|
* @param mode 当前的ActionMode对象
|
|
|
* @param position 发生变化的项位置
|
|
|
* @param id 项ID
|
|
|
* @param checked 新的选中状态
|
|
|
*/
|
|
|
public void onItemCheckedStateChanged(ActionMode mode, int position, long id,
|
|
|
boolean checked) {
|
|
|
mNotesListAdapter.setCheckedItem(position, checked); // 更新适配器中对应项的选中状态
|
|
|
updateMenu(); // 刷新菜单显示
|
|
|
}
|
|
|
/**
|
|
|
* 处理上下文菜单项点击事件
|
|
|
* @param item 被点击的菜单项
|
|
|
* @return boolean 返回true表示事件已处理
|
|
|
*/
|
|
|
public boolean onMenuItemClick(@NonNull MenuItem item) {
|
|
|
// 检查当前是否有选中的笔记
|
|
|
if (mNotesListAdapter.getSelectedCount() == 0) {
|
|
|
// 显示"未选中任何项"的提示
|
|
|
Toast.makeText(NotesListActivity.this,
|
|
|
getString(R.string.menu_select_none),
|
|
|
Toast.LENGTH_SHORT).show();
|
|
|
return true; // 事件已处理
|
|
|
}
|
|
|
|
|
|
// 根据点击的菜单项ID执行不同操作
|
|
|
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,
|
|
|
(dialog, 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表示事件已处理
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 新建笔记按钮的触摸事件监听器
|
|
|
* 处理透明区域的特殊点击事件分发
|
|
|
*/
|
|
|
private class NewNoteOnTouchListener implements OnTouchListener {
|
|
|
|
|
|
/**
|
|
|
* 处理触摸事件
|
|
|
* @param v 被触摸的视图(新建笔记按钮)
|
|
|
* @param event 触摸事件对象
|
|
|
* @return boolean 返回true表示事件已处理,false继续传递
|
|
|
*/
|
|
|
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(); // 按钮高度
|
|
|
|
|
|
// 计算按钮在屏幕中的起始Y坐标(屏幕底部向上偏移按钮高度)
|
|
|
int start = screenHeight - newNoteViewHeight;
|
|
|
// 计算触摸事件的绝对Y坐标(相对于屏幕)
|
|
|
int eventY = start + (int) event.getY();
|
|
|
|
|
|
// 如果是子文件夹模式,需要减去标题栏高度
|
|
|
if (mState == ListEditState.SUB_FOLDER) {
|
|
|
eventY -= mTitleBar.getHeight();
|
|
|
start -= mTitleBar.getHeight();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* HACK: 处理按钮透明区域的点击事件分发
|
|
|
* 透明区域定义为:y < -0.12x + 94(基于按钮局部坐标系)
|
|
|
* 94表示透明区域的最大高度(像素)
|
|
|
* 注意:如果按钮背景改变,这个公式需要相应调整
|
|
|
*/
|
|
|
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) {
|
|
|
// 计算Y坐标偏移量并更新分发坐标
|
|
|
mDispatchY += (int) event.getY() - mOriginY;
|
|
|
event.setLocation(event.getX(), mDispatchY);
|
|
|
return mNotesListView.dispatchTouchEvent(event); // 继续分发移动事件
|
|
|
}
|
|
|
break;
|
|
|
}
|
|
|
default: { // 处理ACTION_UP和ACTION_CANCEL
|
|
|
if (mDispatch) {
|
|
|
event.setLocation(event.getX(), mDispatchY); // 保持坐标一致性
|
|
|
mDispatch = false; // 重置分发标志
|
|
|
return mNotesListView.dispatchTouchEvent(event); // 分发抬起/取消事件
|
|
|
}
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
return false; // 不处理事件,继续传递
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 启动异步查询笔记列表
|
|
|
* 根据当前文件夹ID决定查询条件
|
|
|
*/
|
|
|
private void startAsyncNotesListQuery() {
|
|
|
// 选择查询条件:如果是根文件夹使用特殊查询,否则使用普通查询
|
|
|
String selection = (mCurrentFolderId == Notes.ID_ROOT_FOLDER) ? ROOT_FOLDER_SELECTION
|
|
|
: NORMAL_SELECTION;
|
|
|
|
|
|
// 执行异步查询
|
|
|
mBackgroundQueryHandler.startQuery(
|
|
|
FOLDER_NOTE_LIST_QUERY_TOKEN, // 查询标识
|
|
|
null, // cookie对象
|
|
|
Notes.CONTENT_NOTE_URI, // 内容URI
|
|
|
NoteItemData.PROJECTION, // 查询列
|
|
|
selection, // WHERE条件
|
|
|
new String[] { String.valueOf(mCurrentFolderId) }, // WHERE参数
|
|
|
NoteColumns.TYPE + " DESC," + NoteColumns.MODIFIED_DATE + " DESC" // 排序
|
|
|
);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 后台查询处理器
|
|
|
* 继承AsyncQueryHandler实现异步数据库操作
|
|
|
*/
|
|
|
private final class BackgroundQueryHandler extends AsyncQueryHandler {
|
|
|
|
|
|
/**
|
|
|
* 构造函数
|
|
|
* @param contentResolver 内容解析器
|
|
|
*/
|
|
|
public BackgroundQueryHandler(ContentResolver contentResolver) {
|
|
|
super(contentResolver); // 调用父类构造函数
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 查询完成回调
|
|
|
* @param token 查询标识
|
|
|
* @param cookie 附加数据
|
|
|
* @param cursor 查询结果游标
|
|
|
*/
|
|
|
@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:
|
|
|
// 其他查询类型不处理
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 显示文件夹选择菜单
|
|
|
* @param cursor 包含文件夹数据的游标
|
|
|
*/
|
|
|
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, (dialog, which) -> {
|
|
|
// 执行批量移动操作
|
|
|
DataUtils.batchMoveToFolder(
|
|
|
mContentResolver,
|
|
|
mNotesListAdapter.getSelectedItemIds(), // 选中笔记ID集合
|
|
|
adapter.getItemId(which) // 目标文件夹ID
|
|
|
);
|
|
|
|
|
|
// 显示操作结果提示
|
|
|
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(); // 显示对话框
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 创建新笔记
|
|
|
* 启动笔记编辑Activity并传递当前文件夹ID
|
|
|
*/
|
|
|
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); // 传递当前文件夹ID
|
|
|
|
|
|
// 启动Activity并等待结果
|
|
|
this.startActivityForResult(intent, REQUEST_CODE_NEW_NODE);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 批量删除选中的笔记项
|
|
|
* 根据同步模式决定直接删除还是移动到回收站
|
|
|
*/
|
|
|
private void batchDelete() {
|
|
|
new AsyncTask<Void, Void, HashSet<AppWidgetAttribute>>() {
|
|
|
@Override
|
|
|
protected HashSet<AppWidgetAttribute> doInBackground(Void... unused) {
|
|
|
// 1. 获取选中笔记关联的小部件集合
|
|
|
HashSet<AppWidgetAttribute> widgets = mNotesListAdapter.getSelectedWidget();
|
|
|
|
|
|
// 2. 根据同步模式选择删除策略
|
|
|
if (!isSyncMode()) {
|
|
|
// 非同步模式:直接删除
|
|
|
if (!DataUtils.batchDeleteNotes(mContentResolver,
|
|
|
mNotesListAdapter.getSelectedItemIds())) {
|
|
|
Log.e(TAG, "Delete notes error, should not happen"); // 错误日志
|
|
|
}
|
|
|
} else {
|
|
|
// 同步模式:移动到回收站
|
|
|
if (!DataUtils.batchMoveToFolder(mContentResolver,
|
|
|
mNotesListAdapter.getSelectedItemIds(), Notes.ID_TRASH_FOLER)) {
|
|
|
Log.e(TAG, "Move notes to trash folder error, should not happen");
|
|
|
}
|
|
|
}
|
|
|
return widgets; // 返回需要更新的小部件集合
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
protected void onPostExecute(HashSet<AppWidgetAttribute> widgets) {
|
|
|
// 3. 删除完成后更新关联的小部件
|
|
|
if (widgets != null) {
|
|
|
for (AppWidgetAttribute widget : widgets) {
|
|
|
// 检查小部件ID和类型是否有效
|
|
|
if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID &&
|
|
|
widget.widgetType.get() != Notes.TYPE_WIDGET_INVALIDE) {
|
|
|
updateWidget(widget.widgetId, widget.widgetType.get()); // 更新小部件显示
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
mModeCallBack.finishActionMode(); // 结束多选模式
|
|
|
}
|
|
|
}.execute(); // 启动异步任务
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 删除指定文件夹
|
|
|
* @param folderId 要删除的文件夹ID
|
|
|
*/
|
|
|
private void deleteFolder(long folderId) {
|
|
|
// 检查是否是根文件夹(不允许删除)
|
|
|
if (folderId == Notes.ID_ROOT_FOLDER) {
|
|
|
Log.e(TAG, "Wrong folder id, should not happen " + folderId);
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// 1. 准备要删除的文件夹ID集合
|
|
|
HashSet<Long> ids = new HashSet<>();
|
|
|
ids.add(folderId);
|
|
|
|
|
|
// 2. 获取文件夹关联的小部件
|
|
|
HashSet<AppWidgetAttribute> widgets = DataUtils.getFolderNoteWidget(mContentResolver, folderId);
|
|
|
|
|
|
// 3. 根据同步模式选择删除策略
|
|
|
if (!isSyncMode()) {
|
|
|
DataUtils.batchDeleteNotes(mContentResolver, ids); // 直接删除
|
|
|
} else {
|
|
|
DataUtils.batchMoveToFolder(mContentResolver, ids, Notes.ID_TRASH_FOLER); // 移动到回收站
|
|
|
}
|
|
|
|
|
|
// 4. 更新关联的小部件
|
|
|
if (widgets != null) {
|
|
|
for (AppWidgetAttribute widget : widgets) {
|
|
|
if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID &&
|
|
|
widget.widgetType.get() != Notes.TYPE_WIDGET_INVALIDE) {
|
|
|
updateWidget(widget.widgetId, widget.widgetType.get());
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 打开笔记查看/编辑界面
|
|
|
* @param data 要打开的笔记数据
|
|
|
*/
|
|
|
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
|
|
|
startActivityForResult(intent, REQUEST_CODE_OPEN_NODE); // 启动编辑界面
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 打开文件夹显示其内容
|
|
|
* @param data 要打开的文件夹数据
|
|
|
*/
|
|
|
private void openFolder(NoteItemData data) {
|
|
|
// 1. 更新当前文件夹ID并查询内容
|
|
|
mCurrentFolderId = data.getId();
|
|
|
startAsyncNotesListQuery();
|
|
|
|
|
|
// 2. 根据文件夹类型设置状态
|
|
|
if (data.getId() == Notes.ID_CALL_RECORD_FOLDER) {
|
|
|
mState = ListEditState.CALL_RECORD_FOLDER;
|
|
|
mAddNewNote.setVisibility(View.GONE); // 通话记录文件夹隐藏新建按钮
|
|
|
} else {
|
|
|
mState = ListEditState.SUB_FOLDER;
|
|
|
}
|
|
|
|
|
|
// 3. 更新标题栏显示
|
|
|
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); // 显示标题栏
|
|
|
}
|
|
|
|
|
|
// 处理新建笔记按钮点击
|
|
|
public void onClick(View v) {
|
|
|
if (v.getId() == R.id.btn_new_note) {
|
|
|
createNewNote(); // 调用新建笔记方法
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 强制显示软键盘
|
|
|
*/
|
|
|
private void showSoftInput() {
|
|
|
InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
|
|
|
if (imm != null) {
|
|
|
imm.toggleSoftInput(InputMethodManager.SHOW_FORCED, 0); // 强制显示
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 隐藏软键盘
|
|
|
* @param view 当前焦点视图
|
|
|
*/
|
|
|
private void hideSoftInput(View view) {
|
|
|
InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
|
|
|
imm.hideSoftInputFromWindow(view.getWindowToken(), 0); // 根据窗口token隐藏
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 显示创建/修改文件夹对话框
|
|
|
* @param create true表示创建,false表示修改
|
|
|
*/
|
|
|
private void showCreateOrModifyFolderDialog(final boolean create) {
|
|
|
// 1. 初始化对话框
|
|
|
final AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
|
|
View view = LayoutInflater.from(this).inflate(R.layout.dialog_edit_text, null);
|
|
|
final EditText etName = view.findViewById(R.id.et_foler_name);
|
|
|
|
|
|
// 2. 自动弹出软键盘
|
|
|
showSoftInput();
|
|
|
|
|
|
// 3. 根据模式设置初始文本
|
|
|
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));
|
|
|
}
|
|
|
|
|
|
// 4. 设置对话框按钮
|
|
|
builder.setPositiveButton(android.R.string.ok, null); // 先设为null后自定义点击
|
|
|
builder.setNegativeButton(android.R.string.cancel, (dialog, which) -> hideSoftInput(etName));
|
|
|
|
|
|
// 5. 显示对话框
|
|
|
final Dialog dialog = builder.setView(view).show();
|
|
|
final Button positive = dialog.findViewById(android.R.id.button1);
|
|
|
|
|
|
// 6. 自定义确认按钮点击逻辑
|
|
|
positive.setOnClickListener(v -> {
|
|
|
hideSoftInput(etName);
|
|
|
String name = etName.getText().toString();
|
|
|
|
|
|
// 检查文件夹名是否已存在
|
|
|
if (DataUtils.checkVisibleFolderName(mContentResolver, name)) {
|
|
|
Toast.makeText(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(); // 关闭对话框
|
|
|
});
|
|
|
|
|
|
// 7. 初始禁用确认按钮(当名称为空时)
|
|
|
if (TextUtils.isEmpty(etName.getText())) {
|
|
|
positive.setEnabled(false);
|
|
|
}
|
|
|
|
|
|
// 8. 添加文本变化监听
|
|
|
etName.addTextChangedListener(new TextWatcher() {
|
|
|
@Override
|
|
|
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
|
|
|
// 文本变化前(无需实现)
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
public void onTextChanged(CharSequence s, int start, int before, int count) {
|
|
|
// 文本变化时启用/禁用确认按钮
|
|
|
positive.setEnabled(!TextUtils.isEmpty(etName.getText()));
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
public void afterTextChanged(Editable s) {
|
|
|
// 文本变化后(无需实现)
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// 处理返回键按下事件
|
|
|
@Override
|
|
|
public void onBackPressed() {
|
|
|
switch (mState) {
|
|
|
case SUB_FOLDER:
|
|
|
// 从子文件夹返回根目录
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER;
|
|
|
mState = ListEditState.NOTE_LIST;
|
|
|
startAsyncNotesListQuery(); // 重新加载根目录列表
|
|
|
mTitleBar.setVisibility(View.GONE); // 隐藏标题栏
|
|
|
break;
|
|
|
case CALL_RECORD_FOLDER:
|
|
|
// 从通话记录文件夹返回根目录
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER;
|
|
|
mState = ListEditState.NOTE_LIST;
|
|
|
mAddNewNote.setVisibility(View.VISIBLE); // 显示新建笔记按钮
|
|
|
mTitleBar.setVisibility(View.GONE); // 隐藏标题栏
|
|
|
startAsyncNotesListQuery(); // 重新加载根目录列表
|
|
|
break;
|
|
|
case NOTE_LIST:
|
|
|
super.onBackPressed(); // 默认返回行为
|
|
|
break;
|
|
|
default:
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 更新指定的小部件
|
|
|
private void updateWidget(int appWidgetId, int appWidgetType) {
|
|
|
// 创建小部件更新Intent
|
|
|
Intent intent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
|
|
|
|
|
|
// 根据小部件类型设置对应的Provider类
|
|
|
if (appWidgetType == Notes.TYPE_WIDGET_2X) {
|
|
|
intent.setClass(this, NoteWidgetProvider_2x.class);
|
|
|
} else if (appWidgetType == Notes.TYPE_WIDGET_4X) {
|
|
|
intent.setClass(this, NoteWidgetProvider_4x.class);
|
|
|
} else {
|
|
|
Log.e(TAG, "Unsupported widget type");
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// 设置要更新的小部件ID数组
|
|
|
intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, new int[] {appWidgetId});
|
|
|
|
|
|
sendBroadcast(intent); // 发送广播更新小部件
|
|
|
setResult(RESULT_OK, intent); // 设置操作结果为成功
|
|
|
}
|
|
|
|
|
|
// 文件夹长按菜单创建监听器
|
|
|
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); // 添加重命名菜单项
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// 上下文菜单关闭时调用
|
|
|
@Override
|
|
|
public void onContextMenuClosed(@NonNull Menu menu) {
|
|
|
if (mNotesListView != null) {
|
|
|
mNotesListView.setOnCreateContextMenuListener(null); // 移除菜单监听器
|
|
|
}
|
|
|
super.onContextMenuClosed(menu);
|
|
|
}
|
|
|
|
|
|
// 处理上下文菜单项选择
|
|
|
@Override
|
|
|
public boolean onContextItemSelected(@NonNull MenuItem item) {
|
|
|
if (mFocusNoteDataItem == null) {
|
|
|
Log.e(TAG, "The long click data item is null");
|
|
|
return 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,
|
|
|
(dialog, 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;
|
|
|
}
|
|
|
|
|
|
// 准备选项菜单
|
|
|
@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;
|
|
|
}
|
|
|
|
|
|
// 处理选项菜单项选择
|
|
|
@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;
|
|
|
}
|
|
|
|
|
|
// 处理搜索请求
|
|
|
@Override
|
|
|
public boolean onSearchRequested() {
|
|
|
startSearch(null, false, null /* appData */, false);
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 将笔记导出为文本文件的后台操作方法
|
|
|
* 1. 使用异步任务在后台执行导出操作
|
|
|
* 2. 根据导出结果显示相应的提示对话框
|
|
|
*/
|
|
|
private void exportNoteToText() {
|
|
|
// 获取BackupUtils单例实例,用于执行导出操作
|
|
|
final BackupUtils backup = BackupUtils.getInstance(NotesListActivity.this);
|
|
|
|
|
|
// 创建异步任务执行导出操作(参数Void表示不需要进度更新)
|
|
|
new AsyncTask<Void, Void, Integer>() {
|
|
|
|
|
|
/**
|
|
|
* 后台执行方法
|
|
|
* @param unused 可变参数(此处不需要)
|
|
|
* @return 导出操作的结果状态码
|
|
|
*/
|
|
|
@Override
|
|
|
protected Integer doInBackground(Void... unused) {
|
|
|
// 调用BackupUtils执行实际导出操作,返回操作结果状态码
|
|
|
return backup.exportToText();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 后台任务完成后在主线程执行
|
|
|
* @param result 导出操作结果状态码
|
|
|
*/
|
|
|
|
|
|
@Override
|
|
|
protected void onPostExecute(Integer result) {
|
|
|
// 处理SD卡未挂载情况
|
|
|
if (result == BackupUtils.STATE_SD_CARD_UNMOUONTED) {
|
|
|
// 创建错误对话框构建器
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
// 设置对话框标题(导出失败)
|
|
|
builder.setTitle(NotesListActivity.this.getString(R.string.failed_sdcard_export));
|
|
|
// 设置错误消息(SD卡未挂载)
|
|
|
builder.setMessage(NotesListActivity.this.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(NotesListActivity.this.getString(R.string.success_sdcard_export));
|
|
|
// 设置成功消息(包含导出文件路径信息)
|
|
|
builder.setMessage(NotesListActivity.this.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(NotesListActivity.this.getString(R.string.failed_sdcard_export));
|
|
|
// 设置错误消息(导出过程中出现错误)
|
|
|
builder.setMessage(NotesListActivity.this.getString(R.string.error_sdcard_export));
|
|
|
// 添加确认按钮
|
|
|
builder.setPositiveButton(android.R.string.ok, null);
|
|
|
// 显示对话框
|
|
|
builder.show();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
}.execute(); // 立即执行异步任务
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 检查当前是否处于同步模式
|
|
|
* @return true表示已设置同步账户(同步模式),false表示未设置(本地模式)
|
|
|
*/
|
|
|
private boolean isSyncMode() {
|
|
|
// 获取同步账户名并检查是否为空(去除前后空格)
|
|
|
return !NotesPreferenceActivity.getSyncAccountName(this).trim().isEmpty();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 启动设置Activity
|
|
|
* 优先使用父Activity启动(如果存在),否则使用当前Activity
|
|
|
*/
|
|
|
private void startPreferenceActivity() {
|
|
|
// 获取父Activity(FragmentActivity情况下),不存在则使用当前Activity
|
|
|
Activity from = getParent() != null ? getParent() : this;
|
|
|
// 创建跳转到设置页面的Intent
|
|
|
Intent intent = new Intent(from, NotesPreferenceActivity.class);
|
|
|
// 启动Activity(startActivityIfNeeded保证单例)
|
|
|
from.startActivityIfNeeded(intent, -1);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 列表项点击事件监听器
|
|
|
* 处理笔记/文件夹的点击逻辑
|
|
|
*/
|
|
|
private class OnListItemClickListener implements OnItemClickListener {
|
|
|
|
|
|
@Override
|
|
|
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) {
|
|
|
// 调整位置(考虑HeaderViews的影响)
|
|
|
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;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 查询可用目标文件夹(用于移动笔记操作)
|
|
|
* 查询条件:
|
|
|
* 1. 文件夹类型
|
|
|
* 2. 非回收站文件夹
|
|
|
* 3. 非当前所在文件夹
|
|
|
* 特殊处理:在根目录状态下包含根文件夹选项
|
|
|
*/
|
|
|
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, // 内容URI
|
|
|
FoldersListAdapter.PROJECTION, // 查询列
|
|
|
selection, // 查询条件
|
|
|
new String[] { // 查询参数值
|
|
|
String.valueOf(Notes.TYPE_FOLDER), // 文件夹类型
|
|
|
String.valueOf(Notes.ID_TRASH_FOLER), // 排除回收站
|
|
|
String.valueOf(mCurrentFolderId) // 排除当前文件夹
|
|
|
},
|
|
|
NoteColumns.MODIFIED_DATE + " DESC" // 按修改时间降序
|
|
|
);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 列表项长按事件处理
|
|
|
* @return true表示已处理事件,false继续传递事件
|
|
|
*/
|
|
|
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; // 允许事件继续传递
|
|
|
}
|
|
|
} |