/* * 版权所有 (c) 2010-2011,The MiCode 开源社区 (www.micode.net) * * 本软件根据 Apache 许可证 2.0 版("许可证")发布; * 除非符合许可证,否则不得使用此文件。 * 您可以在以下网址获取许可证副本: * * http://www.apache.org/licenses/LICENSE-2.0 * * 除非法律要求或书面同意,软件 * 根据许可证分发的内容按"原样"提供, * 不附带任何明示或暗示的保证或条件。 * 请参阅许可证,了解有关权限和限制的具体语言。 */ 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; /** * 笔记列表主活动 * 功能: * 1. 展示笔记、文件夹、通话记录的列表界面 * 2. 支持多选模式下的批量操作(删除、移动) * 3. 处理文件夹的创建、重命名、删除 * 4. 集成同步服务(GTaskSyncService)和桌面小部件更新 * 5. 实现数据备份与恢复(导出为文本) * 6. 管理不同层级的列表状态(根目录、子文件夹、通话记录文件夹) */ 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; // 文件夹删除菜单项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, mDispatchY; // 触摸坐标记录 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; // 长按选中的笔记/文件夹数据 // 查询条件:普通列表(子文件夹下的项) 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); if (in != null) { BufferedReader br = new BufferedReader(new InputStreamReader(in)); 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 { DataUtils.closeQuietly(in); // 关闭流工具方法 } // 创建引导笔记 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(); // 标记已创建引导笔记 } } } @Override protected void onStart() { super.onStart(); startAsyncNotesListQuery(); // 启动异步查询加载列表数据 } /** * 初始化界面元素和适配器 */ private void initResources() { mContentResolver = getContentResolver(); mBackgroundQueryHandler = new BackgroundQueryHandler(mContentResolver); mCurrentFolderId = Notes.ID_ROOT_FOLDER; // 初始化为根目录 // 初始化列表视图 mNotesListView = (ListView) findViewById(R.id.notes_list); mNotesListView.addFooterView(LayoutInflater.from(this).inflate(R.layout.note_list_footer, null)); mNotesListView.setOnItemClickListener(new OnListItemClickListener()); mNotesListView.setOnItemLongClickListener(this); // 设置适配器 mNotesListAdapter = new NotesListAdapter(this); mNotesListView.setAdapter(mNotesListAdapter); // 新建笔记按钮 mAddNewNote = (Button) findViewById(R.id.btn_new_note); mAddNewNote.setOnClickListener(this); mAddNewNote.setOnTouchListener(new NewNoteOnTouchListener()); // 自定义触摸事件处理 // 标题栏和状态初始化 mTitleBar = (TextView) findViewById(R.id.tv_title_bar); mState = ListEditState.NOTE_LIST; mModeCallBack = new ModeCallback(); // 多选模式回调实例 } /** * 多选模式回调类(处理ActionMode相关逻辑) */ private class ModeCallback implements ListView.MultiChoiceModeListener, OnMenuItemClickListener { private DropdownMenu mDropDownMenu; // 下拉菜单 private ActionMode mActionMode; // 操作模式 // 创建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); boolean isCallRecordFolder = mFocusNoteDataItem.getParentId() == Notes.ID_CALL_RECORD_FOLDER; boolean hasUserFolders = DataUtils.getUserFolderCount(mContentResolver) > 0; mMoveMenu.setVisible(!isCallRecordFolder && hasUserFolders); if (mMoveMenu.isVisible()) { mMoveMenu.setOnMenuItemClickListener(this); // 移动菜单项监听 } // 设置自定义ActionMode视图(包含下拉菜单) View customView = LayoutInflater.from(NotesListActivity.this).inflate( R.layout.note_list_dropdown_menu, null); mode.setCustomView(customView); mDropDownMenu = new DropdownMenu(NotesListActivity.this, (Button) customView.findViewById(R.id.selection_menu), R.menu.note_list_dropdown); mDropDownMenu.setOnDropdownMenuItemClickListener(item -> { mNotesListAdapter.selectAll(!mNotesListAdapter.isAllSelected()); // 切换全选状态 updateMenu(); // 更新菜单显示 return true; }); mActionMode = mode; mNotesListAdapter.setChoiceMode(true); // 启用多选模式 mNotesListView.setLongClickable(false); // 禁用长按事件 mAddNewNote.setVisibility(View.GONE); // 隐藏新建按钮 return true; } // 更新菜单显示(选中数量和全选状态) private void updateMenu() { int selectedCount = mNotesListAdapter.getSelectedCount(); mDropDownMenu.setTitle(getString(R.string.menu_select_title, selectedCount)); MenuItem selectAllItem = mDropDownMenu.findItem(R.id.action_select_all); if (selectAllItem != null) { boolean isAllSelected = mNotesListAdapter.isAllSelected(); selectAllItem.setChecked(isAllSelected); selectAllItem.setTitle(isAllSelected ? R.string.menu_deselect_all : R.string.menu_select_all); } } // 处理菜单项点击(删除、移动) public boolean onMenuItemClick(MenuItem item) { if (mNotesListAdapter.getSelectedCount() == 0) { Toast.makeText(this, R.string.menu_select_none, Toast.LENGTH_SHORT).show(); return true; } switch (item.getItemId()) { case R.id.delete: // 删除选中项 showDeleteConfirmationDialog(); break; case R.id.move: // 移动选中项到目标文件夹 startQueryDestinationFolders(); break; } return true; } // 省略其他MultiChoiceModeListener接口方法(见完整代码) } /** * 新建笔记按钮触摸事件处理器(处理透明区域的事件分发) */ private class NewNoteOnTouchListener implements OnTouchListener { public boolean onTouch(View v, MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: { // 计算坐标并判断是否触发事件分发(按钮透明区域逻辑) Display display = getWindowManager().getDefaultDisplay(); int screenHeight = display.getHeight(); int buttonHeight = mAddNewNote.getHeight(); int startY = screenHeight - buttonHeight; int eventY = startY + (int) event.getY(); // 调整子文件夹状态下的坐标(减去标题栏高度) if (mState == ListEditState.SUB_FOLDER) { eventY -= mTitleBar.getHeight(); startY -= mTitleBar.getHeight(); } // 透明区域判断公式(根据UI设计硬编码,需与按钮背景匹配) if (event.getY() < (-0.12 * event.getX() + 94)) { View lastItem = mNotesListView.getChildAt(mNotesListView.getChildCount() - 1 - mNotesListView.getFooterViewsCount()); if (lastItem != null && lastItem.getBottom() > startY && lastItem.getTop() < (startY + 94)) { mOriginY = (int) event.getY(); mDispatchY = eventY; event.setLocation(event.getX(), mDispatchY); mDispatch = true; return mNotesListView.dispatchTouchEvent(event); // 分发事件到列表视图 } } break; } // 省略其他触摸事件处理(见完整代码) } return false; } } /** * 启动异步查询加载列表数据 */ private void startAsyncNotesListQuery() { String selection = (mCurrentFolderId == Notes.ID_ROOT_FOLDER) ? ROOT_FOLDER_SELECTION : NORMAL_SELECTION; 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"); // 按类型降序(文件夹优先)、修改时间降序排序 } /** * 后台查询处理器(处理Cursor加载回调) */ private final class BackgroundQueryHandler extends AsyncQueryHandler { public BackgroundQueryHandler(ContentResolver contentResolver) { super(contentResolver); } @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); // 显示文件夹选择菜单 } break; } } } /** * 显示文件夹选择菜单(用于移动操作) */ private void showFolderListMenu(Cursor cursor) { AlertDialog.Builder builder = new AlertDialog.Builder(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(), adapter.getItemId(which)); Toast.makeText(this, getString(R.string.format_move_notes_to_folder, mNotesListAdapter.getSelectedCount(), adapter.getFolderName(this, which)), Toast.LENGTH_SHORT).show(); mModeCallBack.finishActionMode(); // 结束多选模式 }); builder.show(); } /** * 新建笔记 */ 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); startActivityForResult(intent, REQUEST_CODE_NEW_NODE); } /** * 批量删除选中项(异步处理,支持同步模式下移动到回收站) */ private void batch