/* * 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. */ public class NotesListActivity extends Activity implements OnClickListener, OnItemLongClickListener { // 定义查询标记 private static final int FOLDER_NOTE_LIST_QUERY_TOKEN = 0; private static final int FOLDER_LIST_QUERY_TOKEN = 1; // 定义菜单项 private static final int MENU_FOLDER_DELETE = 0; private static final int MENU_FOLDER_VIEW = 1; private static final int MENU_FOLDER_CHANGE_NAME = 2; // 定义一个偏好设置键,用于检查用户是否首次使用应用 private static final String PREFERENCE_ADD_INTRODUCTION = "net.micode.notes.introduction"; // 定义一个枚举,用于表示列表的编辑状态 private enum ListEditState { NOTE_LIST, SUB_FOLDER, CALL_RECORD_FOLDER }; // 成员变量声明 private 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; 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; // 当Activity被创建时调用 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.note_list); // 设置布局 initResources(); // 初始化资源 // 插入一个引导说明,当用户首次使用此应用时 setAppInfoFromRawRes(); } // 处理从其他Activity返回的结果 @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (resultCode == RESULT_OK && (requestCode == REQUEST_CODE_OPEN_NODE || requestCode == REQUEST_CODE_NEW_NODE)) { mNotesListAdapter.changeCursor(null); // 更改适配器中的游标 } else { super.onActivityResult(requestCode, resultCode, data); } } // 从原始资源中设置应用信息(引导说明) private void setAppInfoFromRawRes() { // 获取SharedPreferences实例 SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); // 如果用户没有看过引导说明 if (!sp.getBoolean(PREFERENCE_ADD_INTRODUCTION, false)) { StringBuilder sb = new StringBuilder(); InputStream in = null; try { in = getResources().openRawResource(R.raw.introduction); // 打开原始资源文件 if (in != null) { InputStreamReader isr = new InputStreamReader(in); BufferedReader br = new BufferedReader(isr); char[] buf = new char[1024]; int len; while ((len = br.read(buf)) > 0) { sb.append(buf, 0, len); // 读取文件内容到StringBuilder } } 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()); // 保存笔记,并将引导说明已读标志设置为true if (note.saveNote()) { sp.edit().putBoolean(PREFERENCE_ADD_INTRODUCTION, true).commit(); } else { Log.e(TAG, "Save introduction note error"); // 日志记录错误 return; } } } // 当Activity开始与用户交互时调用 @Override protected void onStart() { super.onStart(); startAsyncNotesListQuery(); // 开始异步查询笔记列表 } // 初始化资源的方法 private void initResources() { // 获取内容解析器,用于访问内容提供者提供的数据 mContentResolver = this.getContentResolver(); // 创建后台查询处理器,用于在后台线程中执行查询 mBackgroundQueryHandler = new BackgroundQueryHandler(this.getContentResolver()); // 设置当前文件夹ID为根文件夹ID 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; // 获取标题栏的TextView,可能用于显示当前状态或标题 mTitleBar = (TextView) 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; // 移动菜单项,用于移动笔记到另一个文件夹 private MenuItem mMoveMenu; // 创建操作模式时的回调 public boolean onCreateActionMode(ActionMode mode, Menu menu) { // 加载菜单资源 getMenuInflater().inflate(R.menu.note_list_options, menu); // 设置删除菜单项的点击监听器 menu.findItem(R.id.delete).setOnMenuItemClickListener(this); // 根据条件设置移动菜单项的可见性 mMoveMenu = menu.findItem(R.id.move); if (mFocusNoteDataItem.getParentId() == Notes.ID_CALL_RECORD_FOLDER || DataUtils.getUserFolderCount(mContentResolver) == 0) { mMoveMenu.setVisible(false); } else { mMoveMenu.setVisible(true); mMoveMenu.setOnMenuItemClickListener(this); } // 保存操作模式实例 mActionMode = mode; // 设置适配器为多选模式 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, (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; } }); 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); } } } } // 这是一个回调接口的实现,用于处理ActionMode生命周期中的事件 public boolean onPrepareActionMode(ActionMode mode, Menu menu) { // 准备ActionMode时调用,这里只是简单地返回false,表示没有特殊准备 return false; } // 当ActionMode中的菜单项被点击时调用 public boolean onActionItemClicked(ActionMode mode, MenuItem item) { // 这里只是简单地返回false,表示没有处理点击事件 return false; } // 当ActionMode被销毁时调用 public void onDestroyActionMode(ActionMode mode) { // 取消列表适配器的选择模式 mNotesListAdapter.setChoiceMode(false); // 允许列表视图长按 mNotesListView.setLongClickable(true); // 使添加新笔记的按钮可见 mAddNewNote.setVisibility(View.VISIBLE); } // 结束当前ActionMode public void finishActionMode() { mActionMode.finish(); } // 当列表项的选择状态改变时调用 public void onItemCheckedStateChanged(ActionMode mode, int position, long id, boolean checked) { // 更新列表适配器中对应位置项的选中状态 mNotesListAdapter.setCheckedItem(position, checked); // 更新ActionMode的菜单项(可能根据选中的项数显示或隐藏某些选项) updateMenu(); } // 当ActionMode中的菜单项被点击时调用(这里似乎是另一个版本的onActionItemClicked,可能是特定于某个实现) public boolean onMenuItemClick(MenuItem item) { // 如果没有选中任何项,显示提示信息 if (mNotesListAdapter.getSelectedCount() == 0) { Toast.makeText(NotesListActivity.this, getString(R.string.menu_select_none), Toast.LENGTH_SHORT).show(); return true; } // 根据点击的菜单项执行相应的操作 switch (item.getItemId()) { case R.id.delete: // 显示删除确认对话框 AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this); builder.setTitle(getString(R.string.alert_title_delete)); builder.setIcon(android.R.drawable.ic_dialog_alert); builder.setMessage(getString(R.string.alert_message_delete_notes, mNotesListAdapter.getSelectedCount())); builder.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int which) { batchDelete(); // 执行批量删除操作 } }); builder.setNegativeButton(android.R.string.cancel, null); builder.show(); break; case R.id.move: // 开始查询目标文件夹的操作(可能是移动选中的笔记) startQueryDestinationFolders(); break; default: return false; } 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(); // 如果处于子文件夹状态,需要减去标题栏的高度 if (mState == ListEditState.SUB_FOLDER) { eventY -= mTitleBar.getHeight(); start -= mTitleBar.getHeight(); } // 这是一个特殊的处理,用于处理“添加新笔记”按钮透明部分的触摸事件 // 将触摸事件转发给列表视图,如果触摸位置满足特定条件 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决定使用哪个SQL选择语句 String selection = (mCurrentFolderId == Notes.ID_ROOT_FOLDER) ? ROOT_FOLDER_SELECTION : NORMAL_SELECTION; // 使用BackgroundQueryHandler启动异步查询 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"); } // 一个内部类,继承自AsyncQueryHandler,用于处理异步查询 private final class BackgroundQueryHandler extends AsyncQueryHandler { // 构造函数,传入ContentResolver以访问内容提供者 public BackgroundQueryHandler(ContentResolver contentResolver) { super(contentResolver); } // 当查询完成时调用此方法 @Override protected void onQueryComplete(int token, Object cookie, Cursor cursor) { // 根据查询的token判断是哪一个查询完成了 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: // 未知的token,不做处理 return; } } } // 显示一个包含文件夹列表的对话框,允许用户选择一个文件夹 private void showFolderListMenu(Cursor cursor) { // 创建AlertDialog.Builder AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this); builder.setTitle(R.string.menu_title_select_folder); // 使用查询结果初始化FoldersListAdapter final FoldersListAdapter adapter = new FoldersListAdapter(this, cursor); // 设置适配器,并定义点击事件的处理逻辑 builder.setAdapter(adapter, new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int which) { // 将选中的笔记移动到用户选择的文件夹中 DataUtils.batchMoveToFolder(mContentResolver, mNotesListAdapter.getSelectedItemIds(), adapter.getItemId(which)); // 显示Toast提示移动成功 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(); } // 创建一个新的笔记 private void createNewNote() { // 创建一个Intent以启动NoteEditActivity Intent intent = new Intent(this, NoteEditActivity.class); intent.setAction(Intent.ACTION_INSERT_OR_EDIT); // 将当前文件夹ID作为额外数据传递给NoteEditActivity intent.putExtra(Notes.INTENT_EXTRA_FOLDER_ID, mCurrentFolderId); // 启动NoteEditActivity,并请求结果 this.startActivityForResult(intent, REQUEST_CODE_NEW_NODE); } // 定义一个方法用于批量删除笔记 private void batchDelete() { // 使用AsyncTask在后台线程中执行批量删除操作,以避免阻塞UI线程 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"); } } // 返回选中的笔记小部件集合,用于后续更新 return widgets; } // 在后台操作完成后,在主线程上执行的操作 @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(); // 执行AsyncTask } // 定义一个方法用于删除文件夹 private void deleteFolder(long folderId) { // 检查是否尝试删除根文件夹,这是不允许的 if (folderId == Notes.ID_ROOT_FOLDER) { Log.e(TAG, "Wrong folder id, should not happen " + folderId); return; } // 创建一个集合,用于存储要删除的文件夹ID HashSet ids = new HashSet(); ids.add(folderId); // 获取与要删除的文件夹相关的笔记小部件集合 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); // 更新小部件 } } } } // 定义一个方法用于打开笔记节点(即笔记详情页面) private void openNode(NoteItemData data) { // 创建一个Intent,用于跳转到NoteEditActivity Intent intent = new Intent(this, NoteEditActivity.class); intent.setAction(Intent.ACTION_VIEW); // 设置Action为查看 intent.putExtra(Intent.EXTRA_UID, data.getId()); // 将笔记ID作为额外数据传递给新Activity // 启动新Activity,并请求结果 this.startActivityForResult(intent, REQUEST_CODE_OPEN_NODE); } // 定义一个方法,用于打开指定的文件夹 private void openFolder(NoteItemData data) { // 更新当前文件夹ID mCurrentFolderId = data.getId(); // 开始异步查询笔记列表 startAsyncNotesListQuery(); // 如果打开的是通话记录文件夹 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); // 隐藏指定视图窗口上的软键盘 inputMethodManager.hideSoftInputFromWindow(view.getWindowToken(), 0); } // 显示创建或修改文件夹的对话框 private void showCreateOrModifyFolderDialog(final boolean create) { // 初始化对话框构建器 final AlertDialog.Builder builder = new AlertDialog.Builder(this); // 加载对话框布局 View view = LayoutInflater.from(this).inflate(R.layout.dialog_edit_text, null); // 获取并显示软键盘 final EditText etName = (EditText) view.findViewById(R.id.et_foler_name); showSoftInput(); // 根据是创建还是修改文件夹,设置不同的标题和初始文本 if (!create) { if (mFocusNoteDataItem != null) { etName.setText(mFocusNoteDataItem.getSnippet()); builder.setTitle(getString(R.string.menu_folder_change_name)); } else { Log.e(TAG, "The long click data item is null"); return; } } else { etName.setText(""); builder.setTitle(this.getString(R.string.menu_create_folder)); } // 设置确定和取消按钮 builder.setPositiveButton(android.R.string.ok, null); builder.setNegativeButton(android.R.string.cancel, new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int which) { // 取消时隐藏软键盘 hideSoftInput(etName); } }); // 显示对话框 final Dialog dialog = builder.setView(view).show(); // 获取并设置确定按钮的点击事件 final Button positive = (Button)dialog.findViewById(android.R.id.button1); positive.setOnClickListener(new OnClickListener() { public void onClick(View v) { // 确定时隐藏软键盘 hideSoftInput(etName); // 获取输入的文件夹名 String name = etName.getText().toString(); // 检查文件夹名是否已存在 if (DataUtils.checkVisibleFolderName(mContentResolver, name)) { Toast.makeText(NotesListActivity.this, getString(R.string.folder_exist, name), Toast.LENGTH_LONG).show(); etName.setSelection(0, etName.length()); return; } // 根据是创建还是修改,执行相应的数据库操作 if (!create) { if (!TextUtils.isEmpty(name)) { ContentValues values = new ContentValues(); values.put(NoteColumns.SNIPPET, name); values.put(NoteColumns.TYPE, Notes.TYPE_FOLDER); values.put(NoteColumns.LOCAL_MODIFIED, 1); mContentResolver.update(Notes.CONTENT_NOTE_URI, values, NoteColumns.ID + "=?", new String[] { String.valueOf(mFocusNoteDataItem.getId()) }); } } else if (!TextUtils.isEmpty(name)) { ContentValues values = new ContentValues(); values.put(NoteColumns.SNIPPET, name); values.put(NoteColumns.TYPE, Notes.TYPE_FOLDER); mContentResolver.insert(Notes.CONTENT_NOTE_URI, values); } // 关闭对话框 dialog.dismiss(); } }); // 根据输入框内容启用或禁用确定按钮 if (TextUtils.isEmpty(etName.getText())) { positive.setEnabled(false); } // 添加文本变化监听器,动态启用或禁用确定按钮 etName.addTextChangedListener(new TextWatcher() { public void beforeTextChanged(CharSequence s, int start, int count, int after) { // 不需要处理 } 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() { // 根据当前的状态(mState)执行不同的操作 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; } } // 更新指定ID和类型的小部件 private void updateWidget(int appWidgetId, int appWidgetType) { Intent intent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE); // 创建更新小部件的Intent if (appWidgetType == Notes.TYPE_WIDGET_2X) { // 如果是2x大小的小部件 intent.setClass(this, NoteWidgetProvider_2x.class); // 设置目标为2x小部件的Provider } else if (appWidgetType == Notes.TYPE_WIDGET_4X) { // 如果是4x大小的小部件 intent.setClass(this, NoteWidgetProvider_4x.class); // 设置目标为4x小部件的Provider } else { Log.e(TAG, "Unsupported widget type"); // 不支持的类型,记录错误日志 return; } intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, new int[] { appWidgetId }); // 添加小部件ID到Intent 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) { 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, 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(); if (mState == ListEditState.NOTE_LIST) { getMenuInflater().inflate(R.menu.note_list, menu); // set sync or 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) { 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; } // 重写onOptionsItemSelected方法,用于处理菜单项的点击事件 @Override public boolean onOptionsItemSelected(MenuItem item) { // 根据点击的菜单项ID执行不同的操作 switch (item.getItemId()) { case R.id.menu_new_folder: { // 显示创建或修改文件夹的对话框,传入true表示是创建新文件夹 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; // 表示事件已处理 } // 重写onSearchRequested方法,用于处理搜索请求 @Override public boolean onSearchRequested() { // 发起搜索请求,不指定搜索初始查询、不调用搜索建议接口、不使用应用提供的额外数据 startSearch(null, false, null /* appData */, false); return true; // 表示事件已处理 } // 导出笔记到文本文件的私有方法 private void exportNoteToText() { // 获取BackupUtils实例 final BackupUtils backup = BackupUtils.getInstance(NotesListActivity.this); // 使用AsyncTask在后台线程执行导出操作 new AsyncTask() { @Override protected Integer doInBackground(Void... unused) { // 执行导出操作,并返回结果码 return backup.exportToText(); } @Override protected void onPostExecute(Integer result) { // 根据结果码显示不同的对话框 if (result == BackupUtils.STATE_SD_CARD_UNMOUONTED) { // SD卡未挂载 showSdCardUnmountedDialog(); } else if (result == BackupUtils.STATE_SUCCESS) { // 导出成功 showExportSuccessDialog(backup.getExportedTextFileName(), backup.getExportedTextFileDir()); } else if (result == BackupUtils.STATE_SYSTEM_ERROR) { // 系统错误 showExportFailedDialog(); } } // 省略了对话框的创建和显示代码,以showSdCardUnmountedDialog()、showExportSuccessDialog()、showExportFailedDialog()表示 }.execute(); } // 判断当前是否处于同步模式的私有方法 private boolean isSyncMode() { // 获取同步账户名,如果非空则处于同步模式 return NotesPreferenceActivity.getSyncAccountName(this).trim().length() > 0; } // 跳转到设置界面的私有方法 private void startPreferenceActivity() { // 获取启动Intent的Activity,如果当前Activity有父Activity,则使用父Activity,否则使用当前Activity Activity from = getParent() != null ? getParent() : this; // 创建跳转到设置界面的Intent Intent intent = new Intent(from, NotesPreferenceActivity.class); // 根据需要启动设置界面Activity from.startActivityIfNeeded(intent, -1); } // 内部类OnListItemClickListener实现了OnItemClickListener接口,用于处理列表项的点击事件 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(); // 如果列表处于选择模式 if (mNotesListAdapter.isInChoiceMode()) { // 根据点击的项是否是笔记类型,更新选择状态 if (item.getType() == Notes.TYPE_NOTE) { position = position - mNotesListView.getHeaderViewsCount(); mModeCallBack.onItemCheckedStateChanged(null, position, id, !mNotesListAdapter.isSelectedItem(position)); } return; } // 根据当前状态(主列表、子文件夹、通话记录文件夹)和点击的项类型执行不同的操作 switch (mState) { case NOTE_LIST: // 如果是文件夹或系统项,打开文件夹;如果是笔记项,打开笔记 if (item.getType() == Notes.TYPE_FOLDER || item.getType() == Notes.TYPE_SYSTEM) { openFolder(item); } else if (item.getType() == Notes.TYPE_NOTE) { openNode(item); } else { // 日志记录:主列表中出现了错误的笔记类型 Log.e(TAG, "Wrong note type in NOTE_LIST"); } break; case SUB_FOLDER: case CALL_RECORD_FOLDER: // 如果是笔记项,打开笔记;否则记录错误日志 if (item.getType() == Notes.TYPE_NOTE) { openNode(item); } else { Log.e(TAG, "Wrong note type in SUB_FOLDER"); } break; default: break; } } } // 省略了openFolder、openNode等方法的实现细节 } // 定义一个私有方法,用于启动查询目标文件夹的操作 private void startQueryDestinationFolders() { // 定义一个SQL查询条件字符串,用于筛选特定类型的笔记,排除特定父ID和ID的笔记 String selection = NoteColumns.TYPE + "=? AND " + NoteColumns.PARENT_ID + "<>? AND " + NoteColumns.ID + "<>?"; // 根据当前的状态(mState),如果是笔记列表状态,则使用上述筛选条件; // 如果不是,则添加一个额外的条件,允许查询根文件夹(ID_ROOT_FOLDER) selection = (mState == ListEditState.NOTE_LIST) ? selection: "(" + selection + ") OR (" + NoteColumns.ID + "=" + Notes.ID_ROOT_FOLDER + ")"; // 使用后台查询处理器(mBackgroundQueryHandler)开始执行查询。 // 查询的目标是笔记内容URI(Notes.CONTENT_NOTE_URI), // 投影列(FoldersListAdapter.PROJECTION)指定了查询返回的列, // selection参数是上述定义的SQL查询条件, // selectionArgs是查询条件的参数值数组,包括文件夹类型、垃圾桶文件夹ID和当前文件夹ID, // orderBy参数指定了结果按修改日期降序排列。 mBackgroundQueryHandler.startQuery(FOLDER_LIST_QUERY_TOKEN, null, Notes.CONTENT_NOTE_URI, FoldersListAdapter.PROJECTION, selection, new String[] { String.valueOf(Notes.TYPE_FOLDER), String.valueOf(Notes.ID_TRASH_FOLER), // 注意:这里可能是一个拼写错误,应该是ID_TRASH_FOLDER String.valueOf(mCurrentFolderId) }, NoteColumns.MODIFIED_DATE + " DESC"); } // 定义一个公开方法,处理列表项的长按事件 public boolean onItemLongClick(AdapterView parent, View view, int position, long id) { // 如果被长按的视图是NotesListItem的实例 if (view instanceof NotesListItem) { // 获取被长按的列表项的数据 mFocusNoteDataItem = ((NotesListItem) view).getItemData(); // 如果该项是笔记类型(Notes.TYPE_NOTE),并且当前不在选择模式下 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 { // 如果启动动作模式失败,则记录错误日志 Log.e(TAG, "startActionMode fails"); } } // 如果该项是文件夹类型(Notes.TYPE_FOLDER) else if (mFocusNoteDataItem.getType() == Notes.TYPE_FOLDER) { // 设置上下文菜单创建监听器,用于文件夹的上下文菜单 mNotesListView.setOnCreateContextMenuListener(mFolderOnCreateContextMenuListener); } } // 返回false,表示事件没有被消费(即没有阻止后续的事件处理) return false; }