diff --git a/NotesListActivity.java b/NotesListActivity.java new file mode 100644 index 0000000..1db68d0 --- /dev/null +++ b/NotesListActivity.java @@ -0,0 +1,1219 @@ +/* + * 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; + +// 导入Android框架中的各种类 +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.DialogInterface; // 用于管理对话框的接口 +import android.content.Intent; // 用于启动其他组件 +import android.content.SharedPreferences; // 用于保存和获取应用设置 +import android.database.Cursor; // 用于数据库查询结果的游标 +import android.os.AsyncTask; // 用于后台任务的异步处理 +import android.os.Bundle; // 用于存储Activity的状态 +import android.preference.PreferenceManager; // 管理SharedPreferences的工具类 +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; // 用于适配器视图(如ListView)的操作 +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; // 用于显示短暂消息的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; // 用于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; // 用于解析资源的工具类 +import net.micode.notes.ui.NotesListAdapter.AppWidgetAttribute; // 用于App Widget的适配器 +import net.micode.notes.widget.NoteWidgetProvider_2x; // 用于提供2x大小的笔记小部件 +import net.micode.notes.widget.NoteWidgetProvider_4x; // 用于提供4x大小的笔记小部件 + +import java.io.BufferedReader; // 用于缓冲读取文本数据 +import java.io.IOException; // 用于处理I/O异常 +import java.io.InputStream; // 用于输入流操作 +import java.io.InputStreamReader; // 用于将输入流转换为字符流 +import java.util.HashSet; // 用于存储唯一元素的集合 + +// 在代码中添加详细注释,解释每个功能模块的作用和实现细节。 +// 比如上面引入了很多Android类和自定义类,接下来代码会用到这些类来实现应用程序的具体功能。 + +public class NotesListActivity extends Activity implements OnClickListener, OnItemLongClickListener { + + // 定义了两个常量用于标识查询操作的 token + 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 ListEditState mState; + + // 处理后台查询操作的 Handler + private BackgroundQueryHandler mBackgroundQueryHandler; + + // 用于显示笔记列表的适配器 + private NotesListAdapter mNotesListAdapter; + + // 用于显示笔记列表的 ListView 控件 + private ListView mNotesListView; + + // 新建笔记的按钮 + private Button mAddNewNote; + + // 是否正在调度某些操作的标识 + private boolean mDispatch; + + // 记录初始的 Y 坐标位置 + private int mOriginY; + + // 记录当前调度的 Y 坐标位置 + private int mDispatchY; + + // 标题栏文本 + private TextView mTitleBar; + + // 当前文件夹的 ID + private long mCurrentFolderId; + + // 内容解析器,用于与内容提供者交互 + private ContentResolver mContentResolver; + + // 模式回调,用于响应 UI 操作 + private ModeCallback mModeCallBack; + + // 日志标签 + private static final String TAG = "NotesListActivity"; + + // 设置滚动速率 + public static final int NOTES_LISTVIEW_SCROLL_RATE = 30; + + // 当前聚焦的笔记数据项 + private NoteItemData mFocusNoteDataItem; + + // 用于查询的正常选择条件:父目录 ID + 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(); + + /** + * Insert an introduction when user firstly use this application + */ + 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 { + // 否则,调用父类的 onActivityResult 处理 + super.onActivityResult(requestCode, resultCode, data); + } +} + +private void setAppInfoFromRawRes() { + // 获取应用的 SharedPreferences,用于存储应用的设置 + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + + // 判断是否已经添加过介绍内容,如果添加过则不再执行 + if (!sp.getBoolean(PREFERENCE_ADD_INTRODUCTION, false)) { + // 创建一个 StringBuilder 来存储介绍内容 + StringBuilder sb = new StringBuilder(); + InputStream in = null; + + try { + // 打开 raw 资源文件,获取介绍内容 + 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; + + // 逐行读取文件内容并追加到 StringBuilder 中 + 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) { + // 处理关闭流时的异常 + 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()); + + // 保存介绍笔记,并在 SharedPreferences 中标记已添加介绍内容 + if (note.saveNote()) { + sp.edit().putBoolean(PREFERENCE_ADD_INTRODUCTION, true).commit(); + } else { + // 如果保存笔记失败,输出错误日志 + Log.e(TAG, "Save introduction note error"); + return; + } + } +} + + + @Override + protected void onStart() { + super.onStart(); + startAsyncNotesListQuery(); + } + + private void initResources() { + mContentResolver = this.getContentResolver(); + mBackgroundQueryHandler = new BackgroundQueryHandler(this.getContentResolver()); + mCurrentFolderId = Notes.ID_ROOT_FOLDER; + 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); + 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); + mState = ListEditState.NOTE_LIST; + mModeCallBack = new ModeCallback(); + } + + // ModeCallback 类实现了 MultiChoiceModeListener 和 OnMenuItemClickListener +// 用于支持 ListView 的多选模式以及自定义菜单的行为 +private class ModeCallback implements ListView.MultiChoiceModeListener, OnMenuItemClickListener { + private DropdownMenu mDropDownMenu; // 定义下拉菜单 + private ActionMode mActionMode; // 当前的 ActionMode + private MenuItem mMoveMenu; // 移动菜单项 + + // 当启动 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); + } + + // 设置 ActionMode 实例 + mActionMode = mode; + + // 设置 ListView 为多选模式 + mNotesListAdapter.setChoiceMode(true); + + // 禁止长按列表项时触发点击事件 + mNotesListView.setLongClickable(false); + + // 隐藏“新建笔记”按钮 + mAddNewNote.setVisibility(View.GONE); + + // 加载自定义视图到 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(new PopupMenu.OnMenuItemClickListener() { + public boolean onMenuItemClick(MenuItem item) { + // 切换选择所有或取消选择所有笔记 + mNotesListAdapter.selectAll(!mNotesListAdapter.isAllSelected()); + updateMenu(); // 更新菜单 + return true; + } + }); + + // 返回 true 表示成功创建了 ActionMode + return true; + } + + // 用于更新下拉菜单中的选择状态 + 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); + } + } + } + + + public boolean onPrepareActionMode(ActionMode mode, Menu menu) { + // TODO Auto-generated method stub + return false; + } + + public boolean onActionItemClicked(ActionMode mode, MenuItem item) { + // TODO Auto-generated method stub + return false; + } + + public void onDestroyActionMode(ActionMode mode) { + // 退出 ActionMode 时,禁用选择模式,并恢复原有状态。 + mNotesListAdapter.setChoiceMode(false); // 禁用适配器中的选择模式(即禁用多选) + mNotesListView.setLongClickable(true); // 使 ListView 恢复长按事件的响应 + mAddNewNote.setVisibility(View.VISIBLE); // 恢复显示“添加新笔记”按钮 +} + +public void finishActionMode() { + // 结束当前的 ActionMode,通常是在完成操作后调用 + mActionMode.finish(); // 调用 ActionMode 的 finish 方法来结束当前操作模式 +} + +public void onItemCheckedStateChanged(ActionMode mode, int position, long id, + boolean checked) { + // 在用户选中或取消选择某项时调用 + mNotesListAdapter.setCheckedItem(position, checked); // 更新适配器中选项的选中状态 + updateMenu(); // 更新菜单项(例如更新删除或移动按钮的状态) +} + +public boolean onMenuItemClick(MenuItem item) { + // 当用户点击 ActionMode 菜单中的某项时调用 + if (mNotesListAdapter.getSelectedCount() == 0) { + // 如果没有任何项被选中,显示提示消息 + Toast.makeText(NotesListActivity.this, getString(R.string.menu_select_none), + Toast.LENGTH_SHORT).show(); + return true; // 消费该事件 + } + + // 处理菜单项点击事件 + switch (item.getItemId()) { + case R.id.delete: + // 用户选择删除 + AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this); + builder.setTitle(getString(R.string.alert_title_delete)) // 设置对话框标题 + .setIcon(android.R.drawable.ic_dialog_alert) // 设置对话框图标 + .setMessage(getString(R.string.alert_message_delete_notes, + mNotesListAdapter.getSelectedCount())) // 设置删除提示信息,显示选中项的数量 + .setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { + public void onClick(DialogInterface dialog, int which) { + batchDelete(); // 执行批量删除操作 + } + }) + .setNegativeButton(android.R.string.cancel, null) // 取消按钮 + .show(); // 显示对话框 + break; + case R.id.move: + // 用户选择移动 + startQueryDestinationFolders(); // 启动查询目标文件夹的操作 + break; + default: + return false; // 其他菜单项,返回 false,不处理 + } + return true; // 表示该事件已被处理 + } +} + 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 newNoteViewHeight = mAddNewNote.getHeight(); + int start = screenHeight - newNoteViewHeight; + int eventY = start + (int) event.getY(); + /** + * Minus TitleBar's height + */ + if (mState == ListEditState.SUB_FOLDER) { + eventY -= mTitleBar.getHeight(); + start -= mTitleBar.getHeight(); + } + /** + * HACKME:When click the transparent part of "New Note" button, dispatch + * the event to the list view behind this button. The transparent part of + * "New Note" button could be expressed by formula y=-0.12x+94(Unit:pixel) + * and the line top of the button. The coordinate based on left of the "New + * Note" button. The 94 represents maximum height of the transparent part. + * Notice that, if the background of the button changes, the formula should + * also change. This is very bad, just for the UI designer's strong requirement. + */ + 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(); + mDispatchY = eventY; + event.setLocation(event.getX(), mDispatchY); + mDispatch = true; + return mNotesListView.dispatchTouchEvent(event); + } + } + break; + } + case MotionEvent.ACTION_MOVE: { + if (mDispatch) { + mDispatchY += (int) event.getY() - mOriginY; + 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; + } + + }; + + // 启动异步查询,查询当前文件夹的笔记列表 +private void startAsyncNotesListQuery() { + // 根据当前文件夹ID选择查询条件 + 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) // 当前文件夹ID作为查询参数 + }, + NoteColumns.TYPE + " DESC," + NoteColumns.MODIFIED_DATE + " DESC"); // 按类型和修改日期降序排序 +} + +// 定义异步查询处理类,继承自 AsyncQueryHandler +private final class BackgroundQueryHandler extends AsyncQueryHandler { + + // 构造方法,传入ContentResolver + public BackgroundQueryHandler(ContentResolver contentResolver) { + super(contentResolver); + } + + // 查询完成后的回调方法 + @Override + protected void onQueryComplete(int token, Object cookie, Cursor cursor) { + switch (token) { + // 如果查询的是笔记列表 + case FOLDER_NOTE_LIST_QUERY_TOKEN: + // 将查询结果的Cursor设置到适配器中 + 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; + } + } +} + +// 显示文件夹列表的弹出菜单 +private void showFolderListMenu(Cursor cursor) { + // 创建AlertDialog.Builder来构建对话框 + AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this); + builder.setTitle(R.string.menu_title_select_folder); // 设置对话框标题 + + // 创建一个自定义的Adapter来显示文件夹列表 + final FoldersListAdapter adapter = new FoldersListAdapter(this, cursor); + + // 设置对话框的Adapter和点击事件 + builder.setAdapter(adapter, new DialogInterface.OnClickListener() { + public void onClick(DialogInterface dialog, int which) { + // 执行批量移动笔记到所选文件夹 + DataUtils.batchMoveToFolder(mContentResolver, + mNotesListAdapter.getSelectedItemIds(), adapter.getItemId(which)); + + // 显示提示Toast,告知用户笔记已成功移动 + Toast.makeText( + NotesListActivity.this, + getString(R.string.format_move_notes_to_folder, + mNotesListAdapter.getSelectedCount(), + adapter.getFolderName(NotesListActivity.this, which)), + Toast.LENGTH_SHORT).show(); + + // 结束Action Mode,取消选择模式 + mModeCallBack.finishActionMode(); + } + }); + + // 显示对话框 + builder.show(); +} + +// 创建新的笔记 +private void createNewNote() { + // 创建一个Intent,跳转到NoteEditActivity来编辑或插入新笔记 + Intent intent = new Intent(this, NoteEditActivity.class); + intent.setAction(Intent.ACTION_INSERT_OR_EDIT); // 设置操作类型为插入或编辑 + intent.putExtra(Notes.INTENT_EXTRA_FOLDER_ID, mCurrentFolderId); // 传递当前文件夹ID + this.startActivityForResult(intent, REQUEST_CODE_NEW_NODE); // 启动活动并等待结果 +} + + + private void batchDelete() { + new AsyncTask>() { + // 后台线程任务,执行批量删除操作 + protected HashSet doInBackground(Void... unused) { + // 获取用户选中的小部件 + HashSet 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"); + } + } + // 返回小部件的集合,以便在UI线程更新 + return widgets; + } + + // 在后台任务执行完后更新UI + @Override + protected void onPostExecute(HashSet widgets) { + // 如果有小部件信息,更新相关的小部件 + if (widgets != null) { + for (AppWidgetAttribute widget : widgets) { + // 如果小部件ID有效且类型不是无效类型,则更新小部件 + if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID + && widget.widgetType != Notes.TYPE_WIDGET_INVALIDE) { + updateWidget(widget.widgetId, widget.widgetType); + } + } + } + // 执行完成后,结束操作模式 + mModeCallBack.finishActionMode(); + } + }.execute(); +} +private void deleteFolder(long folderId) { + // 检查文件夹ID是否是根文件夹的ID,根文件夹不能被删除 + if (folderId == Notes.ID_ROOT_FOLDER) { + Log.e(TAG, "Wrong folder id, should not happen " + folderId); // 打印错误日志 + return; // 如果是根文件夹,返回不做任何操作 + } + + // 创建一个HashSet来保存要删除的文件夹ID + HashSet ids = new HashSet(); + ids.add(folderId); + + // 获取与该文件夹相关的所有小部件(AppWidget),可能有些文件夹有附加的小部件 + HashSet 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) { + // 如果小部件ID有效且小部件类型不是无效类型,则更新小部件 + if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID + && widget.widgetType != Notes.TYPE_WIDGET_INVALIDE) { + // 更新小部件(例如:刷新其显示或者移除等操作) + updateWidget(widget.widgetId, widget.widgetType); + } + } + } +} + + // 打开一个特定的NoteItem (笔记项)并启动NoteEditActivity +private void openNode(NoteItemData data) { + // 创建一个Intent,指定目标Activity为NoteEditActivity + Intent intent = new Intent(this, NoteEditActivity.class); + + // 设置Intent的Action为查看 (ACTION_VIEW) + intent.setAction(Intent.ACTION_VIEW); + + // 将NoteItem的ID作为额外数据传递到NoteEditActivity + intent.putExtra(Intent.EXTRA_UID, data.getId()); + + // 启动NoteEditActivity,并等待返回结果 + this.startActivityForResult(intent, REQUEST_CODE_OPEN_NODE); +} + +// 打开一个文件夹,更新当前文件夹ID并查询相关笔记 +private void openFolder(NoteItemData data) { + // 更新当前文件夹ID + mCurrentFolderId = data.getId(); + + // 启动异步笔记列表查询 + startAsyncNotesListQuery(); + + // 如果当前是“通话记录”文件夹,设置不同的状态和UI + if (data.getId() == Notes.ID_CALL_RECORD_FOLDER) { + mState = ListEditState.CALL_RECORD_FOLDER; + mAddNewNote.setVisibility(View.GONE); // 隐藏新建笔记按钮 + } else { + mState = ListEditState.SUB_FOLDER; + } + + // 根据文件夹的ID设置标题栏文字 + 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) { + // 根据点击的视图ID来判断不同的操作 + switch (v.getId()) { + case R.id.btn_new_note: // 如果点击的是新建笔记按钮 + createNewNote(); // 创建一个新的笔记 + break; + default: + break; + } +} + +// 显示软键盘 +private void showSoftInput() { + // 获取输入法管理器服务 + InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE); + + // 如果输入法管理器不为空,强制显示软键盘 + if (inputMethodManager != null) { + inputMethodManager.toggleSoftInput(InputMethodManager.SHOW_FORCED, 0); + } +} + +// 隐藏软键盘 +private void hideSoftInput(View view) { + // 获取输入法管理器服务 + InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE); + + // 隐藏软键盘,传递当前视图的WindowToken + inputMethodManager.hideSoftInputFromWindow(view.getWindowToken(), 0); +} + + + private void showCreateOrModifyFolderDialog(final boolean create) { + // 创建一个AlertDialog.Builder对象,用于构建对话框 + final AlertDialog.Builder builder = new AlertDialog.Builder(this); + + // 加载自定义的布局(dialog_edit_text),并获取其中的EditText控件 + View view = LayoutInflater.from(this).inflate(R.layout.dialog_edit_text, null); + final EditText etName = (EditText) view.findViewById(R.id.et_foler_name); + + // 显示软键盘,准备输入 + showSoftInput(); + + // 如果不是创建文件夹(即是修改文件夹),设置EditText的文本为当前文件夹的名称 + 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 { + // 如果是创建文件夹,设置EditText为空 + 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(); + + // 获取对话框的确认按钮,并为其设置点击事件 + final Button positive = (Button) dialog.findViewById(android.R.id.button1); + positive.setOnClickListener(new OnClickListener() { + public void onClick(View v) { + hideSoftInput(etName); // 隐藏软键盘 + + // 获取EditText中的文件夹名称 + String name = etName.getText().toString(); + + // 检查文件夹名称是否已存在(DataUtils.checkVisibleFolderName方法进行检查) + 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()) // 使用选中的文件夹ID进行更新 + }); + } + } 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); + } + /** + * When the name edit text is null, disable the positive button + */ + 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 + + } + }); + } + + // 重写返回按钮的行为 +@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 = 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; + } + + // 添加小部件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(Menu menu) { + if (mNotesListView != null) { + mNotesListView.setOnCreateContextMenuListener(null); // 移除菜单监听器 + } + super.onContextMenuClosed(menu); // 调用父类方法 +} + +// 处理上下文菜单项被选中的事件 +@Override +public boolean onContextItemSelected(MenuItem item) { + // 如果当前没有选中笔记项,返回false + if (mFocusNoteDataItem == null) { + Log.e(TAG, "The long click data item is null"); // 记录错误日志 + return false; + } + + // 根据菜单项的ID处理不同的操作 + 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; +} + + @Override +public boolean onPrepareOptionsMenu(Menu menu) { + // 清空菜单中的所有项 + menu.clear(); + + // 根据当前状态(mState)设置不同的菜单 + if (mState == ListEditState.NOTE_LIST) { + // 如果当前状态是 NOTE_LIST,加载 note_list 菜单 + getMenuInflater().inflate(R.menu.note_list, menu); + + // 设置同步(sync)或者取消同步(sync_cancel)菜单项的标题 + menu.findItem(R.id.menu_sync).setTitle( + GTaskSyncService.isSyncing() ? R.string.menu_sync_cancel : R.string.menu_sync); + } else if (mState == ListEditState.SUB_FOLDER) { + // 如果当前状态是 SUB_FOLDER,加载 sub_folder 菜单 + getMenuInflater().inflate(R.menu.sub_folder, menu); + } else if (mState == ListEditState.CALL_RECORD_FOLDER) { + // 如果当前状态是 CALL_RECORD_FOLDER,加载 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; +} + + // 导出笔记到文本文件的函数 +private void exportNoteToText() { + // 获取 BackupUtils 实例 + final BackupUtils backup = BackupUtils.getInstance(NotesListActivity.this); + + // 异步任务:导出笔记 + new AsyncTask() { + + // 在后台线程中执行的任务 + @Override + protected Integer doInBackground(Void... unused) { + // 执行备份操作并返回结果状态 + return backup.exportToText(); + } + + // 在主线程中处理任务完成后的结果 + @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)); // 设置标题 + 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(); // 执行异步任务 +} + +// 判断是否处于同步模式 +private boolean isSyncMode() { + // 如果同步账户名称不为空,表示处于同步模式 + return NotesPreferenceActivity.getSyncAccountName(this).trim().length() > 0; +} + +// 启动设置活动页面 +private void startPreferenceActivity() { + // 获取当前活动的父活动,如果没有父活动,则使用当前活动 + Activity from = getParent() != null ? getParent() : this; + // 创建 Intent,启动设置界面 + Intent intent = new Intent(from, NotesPreferenceActivity.class); + // 启动活动 + from.startActivityIfNeeded(intent, -1); +} + + private class OnListItemClickListener implements OnItemClickListener { + + // 处理列表项点击事件 + public void onItemClick(AdapterView parent, View view, int position, long id) { + + // 确保点击的视图是 NotesListItem 类型 + if (view instanceof NotesListItem) { + NoteItemData item = ((NotesListItem) view).getItemData(); // 获取点击项的 NoteItemData + + // 如果处于多选模式 + if (mNotesListAdapter.isInChoiceMode()) { + // 如果项的类型是 Note 类型 + 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); + } + // 如果项是 Note 类型,打开笔记 + 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: + // 如果项是 Note 类型,打开笔记 + if (item.getType() == Notes.TYPE_NOTE) { + openNode(item); + } else { + Log.e(TAG, "Wrong note type in SUB_FOLDER"); // 如果项类型不正确,输出错误日志 + } + break; + default: + break; // 默认情况下不做任何事情 + } + } + } +} + + + // 开始查询目标文件夹 +private void startQueryDestinationFolders() { + // 设置查询条件:文件类型是文件夹且父文件夹ID不等于给定的ID,ID不等于给定的ID + String selection = NoteColumns.TYPE + "=? AND " + NoteColumns.PARENT_ID + "<>? AND " + NoteColumns.ID + "<>?"; + + // 根据当前状态决定是否需要添加额外的条件 + // 如果当前状态是NOTE_LIST,则使用原始查询条件;否则,添加一个OR条件,查询ID为根文件夹的项 + selection = (mState == ListEditState.NOTE_LIST) ? selection : + "(" + selection + ") OR (" + NoteColumns.ID + "=" + Notes.ID_ROOT_FOLDER + ")"; + + // 启动后台查询,使用FOLDER_LIST_QUERY_TOKEN标识该查询 + mBackgroundQueryHandler.startQuery(FOLDER_LIST_QUERY_TOKEN, + null, // 不需要指定查询的特定ID + Notes.CONTENT_NOTE_URI, // 数据源URI + FoldersListAdapter.PROJECTION, // 查询的列 + selection, // 查询条件 + new String[] { + String.valueOf(Notes.TYPE_FOLDER), // 查询文件夹类型 + String.valueOf(Notes.ID_TRASH_FOLER), // 查询垃圾桶文件夹 + String.valueOf(mCurrentFolderId) // 当前文件夹ID + }, + NoteColumns.MODIFIED_DATE + " DESC"); // 根据修改日期降序排列 +} + +// 处理列表项的长按事件 +public boolean onItemLongClick(AdapterView parent, View view, int position, long id) { + // 确保视图是NotesListItem类型 + if (view instanceof NotesListItem) { + mFocusNoteDataItem = ((NotesListItem) view).getItemData(); // 获取长按项的数据 + + // 判断长按的项是否为笔记类型且没有进入选择模式 + if (mFocusNoteDataItem.getType() == Notes.TYPE_NOTE && !mNotesListAdapter.isInChoiceMode()) { + // 启动操作模式(ActionMode),该模式用于管理选中的项 + if (mNotesListView.startActionMode(mModeCallBack) != null) { + // 在选择模式中选择当前项 + mModeCallBack.onItemCheckedStateChanged(null, position, id, true); + + // 反馈给用户,表示长按事件发生 + mNotesListView.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS); + } else { + // 如果启动ActionMode失败,打印错误日志 + Log.e(TAG, "startActionMode fails"); + } + } + // 判断长按的项是否为文件夹类型 + else if (mFocusNoteDataItem.getType() == Notes.TYPE_FOLDER) { + // 设置文件夹的上下文菜单监听器 + mNotesListView.setOnCreateContextMenuListener(mFolderOnCreateContextMenuListener); + } + } + // 返回false表示该事件没有完全处理,交由其他事件处理器继续处理 + return false; +} +} \ No newline at end of file diff --git a/NotesListAdapter.java b/NotesListAdapter.java new file mode 100644 index 0000000..5492a85 --- /dev/null +++ b/NotesListAdapter.java @@ -0,0 +1,215 @@ +/* + * 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; + +// 导入Android相关的库,用于处理数据库查询和视图的适配 +import android.content.Context; // 用于处理应用程序上下文 +import android.database.Cursor; // 用于数据库查询的游标 +import android.util.Log; // 用于记录日志 +import android.view.View; // 用于视图组件 +import android.view.ViewGroup; // 用于视图容器 +import android.widget.CursorAdapter; // 用于适配器,绑定数据库游标到视图 + +// 导入与笔记相关的类 +import net.micode.notes.data.Notes; // 用于操作笔记数据 + +// 导入集合类,用于数据存储和处理 +import java.util.Collection; // 用于操作一组对象 +import java.util.HashMap; // 用于存储键值对 +import java.util.HashSet; // 用于存储唯一元素集合 +import java.util.Iterator; // 用于遍历集合元素 + + +// 自定义适配器,用于显示笔记列表 +public class NotesListAdapter extends CursorAdapter { + private static final String TAG = "NotesListAdapter"; // 日志标签 + private Context mContext; // 上下文对象,用于访问应用资源 + private HashMap mSelectedIndex; // 用于存储选中项的索引和状态 + private int mNotesCount; // 当前笔记数量 + private boolean mChoiceMode; // 是否处于选择模式 + + // 内部类,用于存储小部件相关的属性 + public static class AppWidgetAttribute { + public int widgetId; // 小部件的 ID + public int widgetType; // 小部件的类型 + }; + + // 构造函数,初始化适配器 + public NotesListAdapter(Context context) { + super(context, null); // 调用父类构造函数,传入上下文和空游标 + mSelectedIndex = new HashMap(); // 初始化存储选中项的 HashMap + mContext = context; // 设置上下文 + mNotesCount = 0; // 初始化笔记数量 + } + + // 创建新视图,当每一项数据被绑定时,调用此方法 + @Override + public View newView(Context context, Cursor cursor, ViewGroup parent) { + return new NotesListItem(context); // 返回一个新的 NotesListItem 视图 + } + + // 绑定数据到视图上 + @Override + public void bindView(View view, Context context, Cursor cursor) { + if (view instanceof NotesListItem) { // 如果视图是 NotesListItem 类型 + // 从游标中提取数据,构建 NoteItemData 对象 + NoteItemData itemData = new NoteItemData(context, cursor); + // 将数据绑定到视图中,并考虑选择模式和选中状态 + ((NotesListItem) view).bind(context, itemData, mChoiceMode, + isSelectedItem(cursor.getPosition())); + } + } + + // 设置指定位置的选中状态 + public void setCheckedItem(final int position, final boolean checked) { + mSelectedIndex.put(position, checked); // 更新选中状态 + notifyDataSetChanged(); // 通知数据发生变化,刷新视图 + } + + // 判断是否处于选择模式 + public boolean isInChoiceMode() { + return mChoiceMode; + } + + // 设置是否进入选择模式 + public void setChoiceMode(boolean mode) { + mSelectedIndex.clear(); // 清除所有选中状态 + mChoiceMode = mode; // 设置选择模式 + } + + // 选择所有项目或取消选择 + public void selectAll(boolean checked) { + Cursor cursor = getCursor(); // 获取当前游标 + for (int i = 0; i < getCount(); i++) { // 遍历所有项 + if (cursor.moveToPosition(i)) { // 移动游标到指定位置 + // 如果该项是笔记类型 + if (NoteItemData.getNoteType(cursor) == Notes.TYPE_NOTE) { + setCheckedItem(i, checked); // 更新选中状态 + } + } + } + } +} + + // 获取已选择的项的ID集合 +public HashSet getSelectedItemIds() { + HashSet itemSet = new HashSet(); // 创建一个HashSet来存储选中的ID + // 遍历所有选中的项的索引 + for (Integer position : mSelectedIndex.keySet()) { + if (mSelectedIndex.get(position) == true) { // 判断该项是否被选中 + Long id = getItemId(position); // 获取该项的ID + if (id == Notes.ID_ROOT_FOLDER) { // 如果ID为根文件夹ID,记录日志并跳过该项 + Log.d(TAG, "Wrong item id, should not happen"); + } else { + itemSet.add(id); // 将ID添加到itemSet中 + } + } + } + return itemSet; // 返回所有选中项的ID集合 +} + +// 获取已选择的小部件(AppWidget)属性集合 +public HashSet getSelectedWidget() { + HashSet itemSet = new HashSet(); // 创建一个HashSet来存储选中的小部件属性 + // 遍历所有选中的项的索引 + for (Integer position : mSelectedIndex.keySet()) { + if (mSelectedIndex.get(position) == true) { // 判断该项是否被选中 + Cursor c = (Cursor) getItem(position); // 获取该项的Cursor对象 + if (c != null) { // 如果Cursor不为空,则处理该项 + AppWidgetAttribute widget = new AppWidgetAttribute(); // 创建一个AppWidgetAttribute对象 + NoteItemData item = new NoteItemData(mContext, c); // 使用Cursor对象初始化NoteItemData + widget.widgetId = item.getWidgetId(); // 获取小部件ID + widget.widgetType = item.getWidgetType(); // 获取小部件类型 + itemSet.add(widget); // 将小部件属性添加到itemSet中 + // 注意:不要在这里关闭Cursor,由适配器负责关闭 + } else { + Log.e(TAG, "Invalid cursor"); // 如果Cursor为空,记录错误日志并返回null + return null; + } + } + } + return itemSet; // 返回所有选中项的小部件属性集合 +} + +// 获取已选中项的数量 +public int getSelectedCount() { + Collection values = mSelectedIndex.values(); // 获取选中项的状态集合 + if (null == values) { + return 0; // 如果没有选中项,返回0 + } + Iterator iter = values.iterator(); + int count = 0; + // 遍历选中项状态,如果为true则计数 + while (iter.hasNext()) { + if (true == iter.next()) { + count++; + } + } + return count; // 返回选中项的数量 +} + +// 判断是否所有项都被选中 +public boolean isAllSelected() { + int checkedCount = getSelectedCount(); // 获取已选中项的数量 + return (checkedCount != 0 && checkedCount == mNotesCount); // 如果已选中项的数量等于总项数,则返回true +} + +// 判断某个位置的项是否被选中 +public boolean isSelectedItem(final int position) { + if (null == mSelectedIndex.get(position)) { + return false; // 如果该位置没有被选中,返回false + } + return mSelectedIndex.get(position); // 返回该位置的选中状态 + + + @Override +protected void onContentChanged() { + // 当内容发生变化时调用,通常用于刷新UI或更新数据 + super.onContentChanged(); + // 重新计算笔记的数量 + calcNotesCount(); +} + +@Override +public void changeCursor(Cursor cursor) { + // 当游标(Cursor)发生变化时调用,通常在数据更新时触发 + super.changeCursor(cursor); + // 重新计算笔记的数量 + calcNotesCount(); +} + +private void calcNotesCount() { + // 初始化笔记数量为0 + mNotesCount = 0; + // 遍历当前的数据项,假设这些数据项是通过Cursor获取的 + for (int i = 0; i < getCount(); i++) { + // 获取当前位置的Cursor对象 + Cursor c = (Cursor) getItem(i); + if (c != null) { + // 如果当前游标存在,检查数据项类型是否为笔记类型 + if (NoteItemData.getNoteType(c) == Notes.TYPE_NOTE) { + // 如果是笔记类型,增加笔记数量 + mNotesCount++; + } + } else { + // 如果游标为null,打印错误日志并返回 + Log.e(TAG, "Invalid cursor"); + return; + } + } + } +} \ No newline at end of file