/*
 * Copyright (c) 2010-2011, The MiCode Open Source Community (www.micode.net)
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *        http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

 package net.micode.notes.ui;

 import android.app.Activity;
 import android.app.AlertDialog;
 import android.app.Dialog;
 import android.appwidget.AppWidgetManager;
 import android.content.AsyncQueryHandler;
 import android.content.ContentResolver;
 import android.content.ContentValues;
 import android.content.Context;
 import android.content.DialogInterface;
 import android.content.Intent;
 import android.content.SharedPreferences;
 import android.database.Cursor;
 import android.os.AsyncTask;
 import android.os.Bundle;
 import android.preference.PreferenceManager;
 import android.text.Editable;
 import android.text.TextUtils;
 import android.text.TextWatcher;
 import android.util.Log;
 import android.view.ActionMode;
 import android.view.ContextMenu;
 import android.view.ContextMenu.ContextMenuInfo;
 import android.view.Display;
 import android.view.HapticFeedbackConstants;
 import android.view.LayoutInflater;
 import android.view.Menu;
 import android.view.MenuItem;
 import android.view.MenuItem.OnMenuItemClickListener;
 import android.view.MotionEvent;
 import android.view.View;
 import android.view.View.OnClickListener;
 import android.view.View.OnCreateContextMenuListener;
 import android.view.View.OnTouchListener;
 import android.view.inputmethod.InputMethodManager;
 import android.widget.AdapterView;
 import android.widget.AdapterView.OnItemClickListener;
 import android.widget.AdapterView.OnItemLongClickListener;
 import android.widget.Button;
 import android.widget.EditText;
 import android.widget.ListView;
 import android.widget.PopupMenu;
 import android.widget.TextView;
 import android.widget.Toast;
 
 import net.micode.notes.R;
 import net.micode.notes.data.Notes;
 import net.micode.notes.data.Notes.NoteColumns;
 import net.micode.notes.gtask.remote.GTaskSyncService;
 import net.micode.notes.model.WorkingNote;
 import net.micode.notes.tool.BackupUtils;
 import net.micode.notes.tool.DataUtils;
 import net.micode.notes.tool.ResourceParser;
 import net.micode.notes.ui.NotesListAdapter.AppWidgetAttribute;
 import net.micode.notes.widget.NoteWidgetProvider_2x;
 import net.micode.notes.widget.NoteWidgetProvider_4x;
 
 import java.io.BufferedReader;
 import java.io.IOException;
 import java.io.InputStream;
 import java.io.InputStreamReader;
 import java.util.HashSet;
 
 /**
  * NotesListActivity 类是一个 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;
     // 用于存储是否添加过介绍的偏好设置键
     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;
     // 显示笔记列表的 ListView
     private ListView mNotesListView;
     // 添加新笔记的按钮
     private Button mAddNewNote;
     // 是否将触摸事件分发到 ListView 的标志
     private boolean mDispatch;
     // 触摸事件的起始 Y 坐标
     private int mOriginY;
     // 触摸事件分发的 Y 坐标
     private int mDispatchY;
     // 标题栏的 TextView
     private TextView mTitleBar;
     // 当前文件夹的 ID
     private long mCurrentFolderId;
     // 内容解析器,用于与 ContentProvider 交互
     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;
 
     /**
      * Activity 创建时调用的方法。
      * 初始化布局和资源,并设置应用信息。
      *
      * @param savedInstanceState 如果 Activity 是重新创建的,则包含之前保存的状态。
      */
     @Override
     protected void onCreate(Bundle savedInstanceState) {
         super.onCreate(savedInstanceState);
         // 设置布局
         setContentView(R.layout.note_list);
         // 初始化资源
         initResources();
 
         /**
          * 当用户首次使用此应用程序时,插入一条介绍信息。
          */
         setAppInfoFromRawRes();
     }
 
     /**
      * 处理 Activity 返回结果的方法。
      * 如果返回结果为 RESULT_OK 且请求码为打开节点或创建新节点,则更新笔记列表适配器的游标。
      *
      * @param requestCode 请求码
      * @param resultCode  返回结果码
      * @param data        返回的意图数据
      */
     @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 sp = PreferenceManager.getDefaultSharedPreferences(this);
         // 检查是否已经添加过介绍信息
         if (!sp.getBoolean(PREFERENCE_ADD_INTRODUCTION, false)) {
             StringBuilder sb = new StringBuilder();
             InputStream in = null;
             try {
                 // 打开原始资源文件
                 in = getResources().openRawResource(R.raw.introduction);
                 if (in != null) {
                     // 创建输入流读取器
                     InputStreamReader isr = new InputStreamReader(in);
                     // 创建缓冲读取器
                     BufferedReader br = new BufferedReader(isr);
                     char[] buf = new char[1024];
                     int len = 0;
                     // 读取文件内容
                     while ((len = br.read(buf)) > 0) {
                         sb.append(buf, 0, len);
                     }
                 } else {
                     // 记录读取介绍文件错误的日志
                     Log.e(TAG, "Read introduction file error");
                     return;
                 }
             } catch (IOException e) {
                 // 打印异常堆栈信息
                 e.printStackTrace();
                 return;
             } finally {
                 if (in != null) {
                     try {
                         // 关闭输入流
                         in.close();
                     } catch (IOException e) {
                         // 打印异常堆栈信息
                         e.printStackTrace();
                     }
                 }
             }
 
             // 创建一个空的笔记
             WorkingNote note = WorkingNote.createEmptyNote(this, Notes.ID_ROOT_FOLDER,
                     AppWidgetManager.INVALID_APPWIDGET_ID, Notes.TYPE_WIDGET_INVALIDE,
                     ResourceParser.RED);
             // 设置笔记的文本内容
             note.setWorkingText(sb.toString());
             if (note.saveNote()) {
                 // 保存笔记成功后,更新共享偏好设置
                 sp.edit().putBoolean(PREFERENCE_ADD_INTRODUCTION, true).commit();
             } else {
                 // 记录保存介绍笔记错误的日志
                 Log.e(TAG, "Save introduction note error");
                 return;
             }
         }
     }
 
     /**
      * Activity 开始时调用的方法。
      * 启动异步笔记列表查询。
      */
     @Override
     protected void onStart() {
         super.onStart();
         // 启动异步笔记列表查询
         startAsyncNotesListQuery();
     }
 
     /**
      * 初始化资源的方法。
      * 初始化 ContentResolver、查询处理程序、ListView、适配器、按钮等。
      */
     private void initResources() {
         // 获取 ContentResolver
         mContentResolver = this.getContentResolver();
         // 创建背景查询处理程序
         mBackgroundQueryHandler = new BackgroundQueryHandler(this.getContentResolver());
         // 设置当前文件夹 ID 为根文件夹 ID
         mCurrentFolderId = Notes.ID_ROOT_FOLDER;
         // 获取笔记列表的 ListView
         mNotesListView = (ListView) findViewById(R.id.notes_list);
         // 为 ListView 添加页脚视图
         mNotesListView.addFooterView(LayoutInflater.from(this).inflate(R.layout.note_list_footer, null),
                 null, false);
         // 设置 ListView 的项点击监听器
         mNotesListView.setOnItemClickListener(new OnListItemClickListener());
         // 设置 ListView 的项长按监听器
         mNotesListView.setOnItemLongClickListener(this);
         // 创建笔记列表适配器
         mNotesListAdapter = new NotesListAdapter(this);
         // 设置 ListView 的适配器
         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;
 
         /**
          * 创建上下文操作模式时调用的方法。
          * 初始化菜单、设置菜单项监听器、更新视图状态。
          *
          * @param mode 上下文操作模式
          * @param menu 上下文操作模式的菜单
          * @return 是否创建成功
          */
         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);
             // 设置 ListView 不可长按
             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);
                 }
             }
         }
 
         /**
          * 准备上下文操作模式时调用的方法。
          * 目前未实现具体逻辑。
          *
          * @param mode 上下文操作模式
          * @param menu 上下文操作模式的菜单
          * @return 是否准备成功
          */
         public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
             // TODO Auto-generated method stub
             return false;
         }
 
         /**
          * 上下文操作模式的菜单项点击时调用的方法。
          * 目前未实现具体逻辑。
          *
          * @param mode 上下文操作模式
          * @param item 被点击的菜单项
          * @return 是否处理成功
          */
         public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
             // TODO Auto-generated method stub
             return false;
         }
 
         /**
          * 销毁上下文操作模式时调用的方法。
          * 恢复视图状态。
          *
          * @param mode 上下文操作模式
          */
         public void onDestroyActionMode(ActionMode mode) {
             // 设置适配器的选择模式为单选
             mNotesListAdapter.setChoiceMode(false);
             // 设置 ListView 可长按
             mNotesListView.setLongClickable(true);
             // 显示添加新笔记的按钮
             mAddNewNote.setVisibility(View.VISIBLE);
         }
 
         /**
          * 结束上下文操作模式的方法。
          */
    /**
     * 结束上下文操作模式的方法
     */
    public void finishActionMode() {
        // 调用 ActionMode 的 finish 方法来结束当前的上下文操作模式
        mActionMode.finish();
    }

    /**
     * 当列表项的选中状态发生改变时调用此方法
     * 
     * @param mode 上下文操作模式
     * @param position 列表项的位置
     * @param id 列表项的 ID
     * @param checked 列表项是否被选中
     */
    public void onItemCheckedStateChanged(ActionMode mode, int position, long id,
            boolean checked) {
        // 调用 NotesListAdapter 的 setCheckedItem 方法,设置指定位置的列表项的选中状态
        mNotesListAdapter.setCheckedItem(position, checked);
        // 调用 updateMenu 方法,更新菜单的显示状态
        updateMenu();
    }

    /**
     * 处理菜单项点击事件的方法
     * 
     * @param item 被点击的菜单项
     * @return 如果事件被处理返回 true,否则返回 false
     */
    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;
        }

        // 根据菜单项的 ID 进行不同的操作
        switch (item.getItemId()) {
            case R.id.delete:
                // 创建一个 AlertDialog 构建器
                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()));
                // 设置对话框的确定按钮,点击确定时调用 batchDelete 方法
                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 {

    /**
     * 处理触摸事件的方法
     * 
     * @param v 触发触摸事件的视图
     * @param event 触摸事件对象
     * @return 如果事件被处理返回 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();
                // 计算按钮顶部的位置
                int start = screenHeight - newNoteViewHeight;
                // 计算触摸事件的 Y 坐标
                int eventY = start + (int) event.getY();
                /**
                 * 如果当前处于子文件夹列表编辑状态,减去标题栏的高度
                 */
                if (mState == ListEditState.SUB_FOLDER) {
                    eventY -= mTitleBar.getHeight();
                    start -= mTitleBar.getHeight();
                }
                /**
                 * 注意:当点击“New Note”按钮的透明部分时,将事件分发到按钮后面的列表视图。
                 * “New Note”按钮的透明部分可以用公式 y = -0.12x + 94(单位:像素)和按钮顶部的线来表示。
                 * 坐标基于“New Note”按钮的左侧。94 表示透明部分的最大高度。
                 * 注意,如果按钮的背景发生变化,公式也应该相应改变。这是为了满足 UI 设计师的强烈要求,不太理想。
                 */
                if (event.getY() < (event.getX() * (-0.12) + 94)) {
                    // 获取列表视图的最后一个非页脚子视图
                    View view = mNotesListView.getChildAt(mNotesListView.getChildCount() - 1
                            - mNotesListView.getFooterViewsCount());
                    // 检查视图是否存在,并且视图的底部位置大于按钮顶部位置,顶部位置小于按钮顶部位置加上 94
                    if (view != null && view.getBottom() > start
                            && (view.getTop() < (start + 94))) {
                        // 记录触摸事件的起始 Y 坐标
                        mOriginY = (int) event.getY();
                        // 记录要分发的 Y 坐标
                        mDispatchY = eventY;
                        // 修改触摸事件的位置
                        event.setLocation(event.getX(), mDispatchY);
                        // 设置分发标志为 true
                        mDispatch = true;
                        // 将触摸事件分发给列表视图
                        return mNotesListView.dispatchTouchEvent(event);
                    }
                }
                break;
            }
            case MotionEvent.ACTION_MOVE: {
                // 如果分发标志为 true
                if (mDispatch) {
                    // 更新要分发的 Y 坐标
                    mDispatchY += (int) event.getY() - mOriginY;
                    // 修改触摸事件的位置
                    event.setLocation(event.getX(), mDispatchY);
                    // 将触摸事件分发给列表视图
                    return mNotesListView.dispatchTouchEvent(event);
                }
                break;
            }
            default: {
                // 如果分发标志为 true
                if (mDispatch) {
                    // 修改触摸事件的位置
                    event.setLocation(event.getX(), mDispatchY);
                    // 设置分发标志为 false
                    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[] {
                // 将当前文件夹 ID 转换为字符串作为查询参数
                String.valueOf(mCurrentFolderId)
            }, NoteColumns.TYPE + " DESC," + NoteColumns.MODIFIED_DATE + " DESC");
}

/**
 * 后台查询处理程序类,继承自 AsyncQueryHandler
 */
private final class BackgroundQueryHandler extends AsyncQueryHandler {
    /**
     * 构造方法,初始化 ContentResolver
     * 
     * @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:
                // 将查询结果的游标传递给 NotesListAdapter
                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;
        }
    }
}

/**
 * 显示文件夹列表菜单的方法
 * 
 * @param cursor 包含文件夹信息的游标
 */
private void showFolderListMenu(Cursor cursor) {
    // 创建一个 AlertDialog 构建器
    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() {

        /**
         * 当点击文件夹列表中的某一项时调用的方法
         * 
         * @param dialog 对话框对象
         * @param which 被点击项的位置
         */
        public void onClick(DialogInterface dialog, int which) {
            // 调用 DataUtils 的 batchMoveToFolder 方法,将选中的笔记移动到指定的文件夹
            DataUtils.batchMoveToFolder(mContentResolver,
                    mNotesListAdapter.getSelectedItemIds(), adapter.getItemId(which));
            // 显示一个短时间的提示信息,告知用户笔记移动成功
            Toast.makeText(
                    NotesListActivity.this,
                    getString(R.string.format_move_notes_to_folder,
                            mNotesListAdapter.getSelectedCount(),
                            adapter.getFolderName(NotesListActivity.this, which)),
                    Toast.LENGTH_SHORT).show();
            // 结束上下文操作模式
            mModeCallBack.finishActionMode();
        }
    });
    // 显示对话框
    builder.show();
}
    /**
     * 创建新笔记的方法。
     * 该方法会创建一个新的意图,启动 NoteEditActivity 来创建新笔记,并传递当前文件夹的 ID。
     */
    private void createNewNote() {
        // 创建一个意图,用于启动 NoteEditActivity
        Intent intent = new Intent(this, NoteEditActivity.class);
        // 设置意图的动作,表明是插入或编辑操作
        intent.setAction(Intent.ACTION_INSERT_OR_EDIT);
        // 向意图中添加额外数据,传递当前文件夹的 ID
        intent.putExtra(Notes.INTENT_EXTRA_FOLDER_ID, mCurrentFolderId);
        // 启动 NoteEditActivity 并等待返回结果
        this.startActivityForResult(intent, REQUEST_CODE_NEW_NODE);
    }

    /**
     * 批量删除选中笔记的方法。
     * 该方法使用异步任务在后台执行删除或移动到回收站的操作,并在操作完成后更新相关小部件。
     */
    private void batchDelete() {
        // 创建一个异步任务,用于在后台执行批量删除或移动到回收站的操作
        new AsyncTask<Void, Void, HashSet<AppWidgetAttribute>>() {
            /**
             * 在后台线程中执行的方法,负责批量删除或移动选中的笔记,并返回相关小部件的集合。
             * @param unused 未使用的参数
             * @return 包含选中笔记关联小部件属性的集合
             */
            protected HashSet<AppWidgetAttribute> doInBackground(Void... unused) {
                // 获取选中笔记关联的小部件集合
                HashSet<AppWidgetAttribute> widgets = mNotesListAdapter.getSelectedWidget();
                // 检查是否处于同步模式
                if (!isSyncMode()) {
                    // 如果不是同步模式,直接删除选中的笔记
                    if (DataUtils.batchDeleteNotes(mContentResolver, mNotesListAdapter
                            .getSelectedItemIds())) {
                    } else {
                        // 记录删除笔记失败的错误日志
                        Log.e(TAG, "Delete notes error, should not happens");
                    }
                } else {
                    // 如果处于同步模式,将选中的笔记移动到回收站文件夹
                    if (!DataUtils.batchMoveToFolder(mContentResolver, mNotesListAdapter
                            .getSelectedItemIds(), Notes.ID_TRASH_FOLER)) {
                        // 记录移动笔记到回收站失败的错误日志
                        Log.e(TAG, "Move notes to trash folder error, should not happens");
                    }
                }
                return widgets;
            }

            /**
             * 在主线程中执行的方法,在后台任务完成后调用,用于更新相关小部件并结束操作模式。
             * @param widgets 包含选中笔记关联小部件属性的集合
             */
            @Override
            protected void onPostExecute(HashSet<AppWidgetAttribute> 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();
    }

    /**
     * 删除指定文件夹的方法。
     * 该方法会根据同步模式的不同,直接删除文件夹或将其移动到回收站,并更新相关小部件。
     * @param folderId 要删除的文件夹的 ID
     */
    private void deleteFolder(long folderId) {
        // 检查文件夹 ID 是否为根文件夹 ID
        if (folderId == Notes.ID_ROOT_FOLDER) {
            // 记录错误日志,因为根文件夹不能被删除
            Log.e(TAG, "Wrong folder id, should not happen " + folderId);
            return;
        }

        // 创建一个包含要删除文件夹 ID 的集合
        HashSet<Long> ids = new HashSet<Long>();
        ids.add(folderId);
        // 获取指定文件夹中笔记关联的小部件集合
        HashSet<AppWidgetAttribute> widgets = DataUtils.getFolderNoteWidget(mContentResolver,
                folderId);
        // 检查是否处于同步模式
        if (!isSyncMode()) {
            // 如果不是同步模式,直接删除文件夹
            DataUtils.batchDeleteNotes(mContentResolver, ids);
        } else {
            // 如果处于同步模式,将文件夹移动到回收站
            DataUtils.batchMoveToFolder(mContentResolver, ids, Notes.ID_TRASH_FOLER);
        }
        // 检查小部件集合是否为空
        if (widgets != null) {
            // 遍历小部件集合
            for (AppWidgetAttribute widget : widgets) {
                // 检查小部件 ID 和类型是否有效
                if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID
                        && widget.widgetType != Notes.TYPE_WIDGET_INVALIDE) {
                    // 更新有效的小部件
                    updateWidget(widget.widgetId, widget.widgetType);
                }
            }
        }
    }

    /**
     * 打开指定笔记的方法。
     * 该方法会创建一个意图,启动 NoteEditActivity 来查看指定笔记。
     * @param data 要打开的笔记的数据对象
     */
    private void openNode(NoteItemData data) {
        // 创建一个意图,用于启动 NoteEditActivity
        Intent intent = new Intent(this, NoteEditActivity.class);
        // 设置意图的动作,表明是查看操作
        intent.setAction(Intent.ACTION_VIEW);
        // 向意图中添加额外数据,传递笔记的 ID
        intent.putExtra(Intent.EXTRA_UID, data.getId());
        // 启动 NoteEditActivity 并等待返回结果
        this.startActivityForResult(intent, REQUEST_CODE_OPEN_NODE);
    }

    /**
     * 打开指定文件夹的方法。
     * 该方法会更新当前文件夹 ID,重新查询该文件夹下的笔记列表,并更新界面状态。
     * @param data 要打开的文件夹的数据对象
     */
    private void openFolder(NoteItemData data) {
        // 更新当前文件夹 ID
        mCurrentFolderId = data.getId();
        // 启动异步查询,获取当前文件夹下的笔记列表
        startAsyncNotesListQuery();
        // 检查文件夹 ID 是否为通话记录文件夹 ID
        if (data.getId() == Notes.ID_CALL_RECORD_FOLDER) {
            // 如果是通话记录文件夹,设置列表编辑状态为通话记录文件夹状态
            mState = ListEditState.CALL_RECORD_FOLDER;
            // 隐藏新建笔记按钮
            mAddNewNote.setVisibility(View.GONE);
        } else {
            // 如果不是通话记录文件夹,设置列表编辑状态为子文件夹状态
            mState = ListEditState.SUB_FOLDER;
        }
        // 检查文件夹 ID 是否为通话记录文件夹 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);
    }

    /**
     * 处理视图点击事件的方法。
     * 根据点击的视图 ID 执行相应的操作。
     * @param v 被点击的视图
     */
    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);
        }
    }

    /**
     * 隐藏软键盘的方法。
     * 该方法会隐藏指定视图关联的软键盘。
     * @param view 关联软键盘的视图
     */
    private void hideSoftInput(View view) {
        // 获取输入法管理器
        InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
        // 隐藏指定视图关联的软键盘
        inputMethodManager.hideSoftInputFromWindow(view.getWindowToken(), 0);
    }

    /**
     * 显示创建或修改文件夹对话框的方法。
     * 根据传入的参数,显示创建文件夹或修改文件夹名称的对话框。
     * @param create 是否为创建文件夹操作,true 表示创建,false 表示修改
     */
    private void showCreateOrModifyFolderDialog(final boolean create) {
        // 创建一个 AlertDialog 构建器
        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() {
            /**
             * 取消按钮点击事件处理方法。
             * 点击取消按钮时,隐藏软键盘。
             * @param dialog 对话框对象
             * @param which 按钮的索引
             */
            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() {
            /**
             * 确定按钮点击事件处理方法。
             * 点击确定按钮时,检查文件夹名称是否已存在,根据操作类型进行创建或修改文件夹的操作。
             * @param v 被点击的视图
             */
            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 对象,用于存储要更新的字段和值
                        ContentValues values = new ContentValues();
                        // 设置文件夹的摘要信息
                        values.put(NoteColumns.SNIPPET, name);
                        // 设置文件夹的类型
                        values.put(NoteColumns.TYPE, Notes.TYPE_FOLDER);
                        // 设置文件夹的本地修改标志
                        values.put(NoteColumns.LOCAL_MODIFIED, 1);
                        // 更新指定 ID 的文件夹信息
                        mContentResolver.update(Notes.CONTENT_NOTE_URI, values, NoteColumns.ID
                                + "=?", new String[] {
                            // 将当前文件夹的 ID 转换为字符串作为查询参数
                            String.valueOf(mFocusNoteDataItem.getId())
                        });
                    }
                } else if (!TextUtils.isEmpty(name)) {
                    // 如果是创建操作,检查文件夹名称是否为空
                    // 创建一个 ContentValues 对象,用于存储要插入的字段和值
                    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() {
            /**
             * 在文本变化之前调用的方法。
             * @param s 变化前的文本
             * @param start 变化的起始位置
             * @param count 变化的字符数
             * @param after 变化后的字符数
             */
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
                // 留空,不需要处理
            }

            /**
             * 在文本变化时调用的方法。
             * 检查输入框中的文本是否为空,并相应地启用或禁用确定按钮。
             * @param s 变化后的文本
             * @param start 变化的起始位置
             * @param before 变化前的字符数
             * @param count 变化的字符数
             */
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                // 检查输入框中的文本是否为空
                if (TextUtils.isEmpty(etName.getText())) {
                    // 如果为空,禁用确定按钮
                    positive.setEnabled(false);
                } else {
                    // 如果不为空,启用确定按钮
                    positive.setEnabled(true);
                }
            }

            /**
             * 在文本变化之后调用的方法。
             * @param s 变化后的可编辑文本
             */
            public void afterTextChanged(Editable s) {
                // 留空,不需要处理
            }
        });
    }

    /**
     * 处理返回键按下事件的方法。
     * 根据当前列表编辑状态,执行相应的操作,如返回上一级文件夹或退出应用。
     */
    @Override
    public void onBackPressed() {
        // 根据当前列表编辑状态进行不同的操作
        switch (mState) {
            case SUB_FOLDER:
                // 如果当前处于子文件夹状态,将当前文件夹 ID 设置为根文件夹 ID
                mCurrentFolderId = Notes.ID_ROOT_FOLDER;
                // 将列表编辑状态设置为笔记列表状态
                mState = ListEditState.NOTE_LIST;
                // 启动异步查询,获取根文件夹下的笔记列表
                startAsyncNotesListQuery();
                // 隐藏标题栏
                mTitleBar.setVisibility(View.GONE);
                break;
            case CALL_RECORD_FOLDER:
                // 如果当前处于通话记录文件夹状态,将当前文件夹 ID 设置为根文件夹 ID
                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;
        }
    }

    /**
     * 更新指定小部件的方法。
     * 该方法会发送一个广播,通知相应的小部件提供者更新小部件。
     * @param appWidgetId 要更新的小部件的 ID
     * @param appWidgetType 要更新的小部件的类型
     */

    /**
 * 更新小部件的方法
 * @param appWidgetId 小部件的ID
 * @param appWidgetType 小部件的类型
 */
private void updateWidget(int appWidgetId, int appWidgetType) {
    // 创建一个用于更新小部件的意图
    Intent intent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
    // 根据小部件类型设置意图的目标类
    if (appWidgetType == Notes.TYPE_WIDGET_2X) {
        // 如果是2x类型的小部件,设置目标类为NoteWidgetProvider_2x
        intent.setClass(this, NoteWidgetProvider_2x.class);
    } else if (appWidgetType == Notes.TYPE_WIDGET_4X) {
        // 如果是4x类型的小部件,设置目标类为NoteWidgetProvider_4x
        intent.setClass(this, NoteWidgetProvider_4x.class);
    } 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() {
    /**
     * 创建上下文菜单时调用的方法
     * @param menu 上下文菜单对象
     * @param v 触发上下文菜单的视图
     * @param menuInfo 上下文菜单信息
     */
    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);
        }
    }
};

/**
 * 上下文菜单关闭时调用的方法
 * @param menu 关闭的上下文菜单
 */
@Override
public void onContextMenuClosed(Menu menu) {
    // 如果笔记列表视图不为空
    if (mNotesListView != null) {
        // 移除笔记列表视图的上下文菜单创建监听器
        mNotesListView.setOnCreateContextMenuListener(null);
    }
    // 调用父类的上下文菜单关闭方法
    super.onContextMenuClosed(menu);
}

/**
 * 上下文菜单项被选中时调用的方法
 * @param item 被选中的菜单项
 * @return 是否处理了该菜单项的选中事件
 */
@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方法打开聚焦的笔记数据项对应的文件夹
            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));
            // 设置对话框的确定按钮,点击确定时调用deleteFolder方法删除文件夹
            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方法显示创建或修改文件夹的对话框
            showCreateOrModifyFolderDialog(false);
            break;
        default:
            break;
    }
    // 返回true表示处理了该菜单项的选中事件
    return true;
}

/**
 * 准备选项菜单时调用的方法
 * @param menu 选项菜单对象
 * @return 是否处理了选项菜单的准备事件
 */
@Override
public boolean onPrepareOptionsMenu(Menu menu) {
    // 清空菜单中的所有菜单项
    menu.clear();
    // 根据当前的列表编辑状态加载不同的菜单布局
    if (mState == ListEditState.NOTE_LIST) {
        // 如果是笔记列表状态,加载笔记列表的菜单布局
        getMenuInflater().inflate(R.menu.note_list, menu);
        // 设置同步菜单项的标题,根据GTaskSyncService是否正在同步来显示不同的标题
        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);
    }
    // 返回true表示处理了选项菜单的准备事件
    return true;
}

/**
 * 选项菜单项被选中时调用的方法
 * @param item 被选中的菜单项
 * @return 是否处理了该菜单项的选中事件
 */
@Override
public boolean onOptionsItemSelected(MenuItem item) {
    // 根据菜单项的ID进行不同的处理
    switch (item.getItemId()) {
        case R.id.menu_new_folder: {
            // 创建新文件夹,调用showCreateOrModifyFolderDialog方法显示创建或修改文件夹的对话框
            showCreateOrModifyFolderDialog(true);
            break;
        }
        case R.id.menu_export_text: {
            // 导出笔记为文本,调用exportNoteToText方法进行导出操作
            exportNoteToText();
            break;
        }
        case R.id.menu_sync: {
            // 同步操作,如果是同步模式
            if (isSyncMode()) {
                // 根据菜单项的标题判断是开始同步还是取消同步
                if (TextUtils.equals(item.getTitle(), getString(R.string.menu_sync))) {
                    // 如果标题是同步,调用GTaskSyncService的startSync方法开始同步
                    GTaskSyncService.startSync(this);
                } else {
                    // 如果标题是取消同步,调用GTaskSyncService的cancelSync方法取消同步
                    GTaskSyncService.cancelSync(this);
                }
            } else {
                // 如果不是同步模式,调用startPreferenceActivity方法启动设置活动
                startPreferenceActivity();
            }
            break;
        }
        case R.id.menu_setting: {
            // 打开设置,调用startPreferenceActivity方法启动设置活动
            startPreferenceActivity();
            break;
        }
        case R.id.menu_new_note: {
            // 创建新笔记,调用createNewNote方法创建新笔记
            createNewNote();
            break;
        }
        case R.id.menu_search:
            // 搜索操作,调用onSearchRequested方法触发搜索请求
            onSearchRequested();
            break;
        default:
            break;
    }
    // 返回true表示处理了该菜单项的选中事件
    return true;
}

/**
 * 触发搜索请求时调用的方法
 * @return 是否处理了搜索请求
 */
@Override
public boolean onSearchRequested() {
    // 启动搜索,传入搜索关键字为空,不使用全局搜索,不携带应用数据,不使用渐进式搜索
    startSearch(null, false, null /* appData */, false);
    // 返回true表示处理了搜索请求
    return true;
}

    /**
 * 将笔记导出为文本文件的方法
 */
private void exportNoteToText() {
    // 获取 BackupUtils 类的单例实例,传入当前 Activity 的上下文
    final BackupUtils backup = BackupUtils.getInstance(NotesListActivity.this);

    // 创建一个异步任务,避免在主线程执行耗时的导出操作
    new AsyncTask<Void, Void, Integer>() {

        /**
         * 此方法在后台线程执行,用于执行实际的导出操作
         * @param unused 未使用的参数
         * @return 导出操作的结果状态码
         */
        @Override
        protected Integer doInBackground(Void... unused) {
            // 调用 BackupUtils 实例的 exportToText 方法进行导出操作,并返回结果状态码
            return backup.exportToText();
        }

        /**
         * 当后台任务执行完成后,此方法在主线程执行,用于处理导出结果
         * @param result 导出操作的结果状态码
         */
        @Override
        protected void onPostExecute(Integer result) {
            // 根据不同的结果状态码,显示不同的提示对话框
            if (result == BackupUtils.STATE_SD_CARD_UNMOUONTED) {
                // 如果 SD 卡未挂载,创建一个警告对话框提示用户
                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));
                // 设置对话框消息为 SD 卡导出错误信息
                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() {
    // 从 NotesPreferenceActivity 中获取同步账户名,去除首尾空格后检查长度是否大于 0
    return NotesPreferenceActivity.getSyncAccountName(this).trim().length() > 0;
}

/**
 * 启动设置偏好活动的方法
 */
private void startPreferenceActivity() {
    // 获取当前 Activity 的父 Activity,如果没有父 Activity 则使用当前 Activity
    Activity from = getParent() != null ? getParent() : this;
    // 创建一个意图,用于启动 NotesPreferenceActivity
    Intent intent = new Intent(from, NotesPreferenceActivity.class);
    // 启动 NotesPreferenceActivity,如果需要的话
    from.startActivityIfNeeded(intent, -1);
}

/**
 * 列表项点击监听器类,实现 OnItemClickListener 接口
 */
private class OnListItemClickListener implements OnItemClickListener {

    /**
     * 当列表项被点击时调用的方法
     * @param parent 列表项所在的适配器视图
     * @param view 被点击的视图
     * @param position 被点击的列表项的位置
     * @param id 被点击的列表项的 ID
     */
    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;
            }
        }
    }

}

/**
 * 开始查询目标文件夹的方法
 */
private void startQueryDestinationFolders() {
    // 构建查询条件,筛选出文件夹类型且不是回收站和当前文件夹的记录
    String selection = NoteColumns.TYPE + "=? AND " + NoteColumns.PARENT_ID + "<>? AND " + NoteColumns.ID + "<>?";
    // 如果当前处于笔记列表状态,使用上述查询条件;否则,添加根文件夹到查询结果中
    selection = (mState == ListEditState.NOTE_LIST) ? selection:
        "(" + selection + ") OR (" + NoteColumns.ID + "=" + Notes.ID_ROOT_FOLDER + ")";

    // 启动异步查询,查询目标文件夹
    mBackgroundQueryHandler.startQuery(FOLDER_LIST_QUERY_TOKEN,
            null,
            Notes.CONTENT_NOTE_URI,
            FoldersListAdapter.PROJECTION,
            selection,
            new String[] {
                    String.valueOf(Notes.TYPE_FOLDER),
                    String.valueOf(Notes.ID_TRASH_FOLER),
                    String.valueOf(mCurrentFolderId)
            },
            NoteColumns.MODIFIED_DATE + " DESC");
}

/**
 * 列表项长按监听器方法
 * @param parent 列表项所在的适配器视图
 * @param view 被长按的视图
 * @param position 被长按的列表项的位置
 * @param id 被长按的列表项的 ID
 * @return 是否处理了长按事件
 */

    /**
 * 处理列表项长按事件的方法
 * 
 * @param parent 触发长按事件的 AdapterView
 * @param view 被长按的视图
 * @param position 被长按的列表项在适配器中的位置
 * @param id 被长按的列表项的 ID
 * @return 表示是否已处理该长按事件,这里返回 false 表示未完全处理,可能会触发其他默认行为
 */
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) {
                // 如果成功启动 ActionMode,则调用 mModeCallBack 的 onItemCheckedStateChanged 方法
                // 将该列表项标记为已选中
                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) {
            // 如果被长按的列表项类型是文件夹类型
            // 设置列表视图的上下文菜单创建监听器为 mFolderOnCreateContextMenuListener
            // 以便在长按文件夹时显示上下文菜单
            mNotesListView.setOnCreateContextMenuListener(mFolderOnCreateContextMenuListener);
        }
    }
    // 返回 false 表示该长按事件未完全处理,可能会触发其他默认行为
    return false;
}
 }