|
|
|
|
@ -149,59 +149,38 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
|
|
|
// 当活动返回结果时,检查结果是否为成功
|
|
|
|
|
// 只有当返回结果为成功,并且请求码为打开或新建笔记时,才进行数据更新
|
|
|
|
|
if (resultCode == RESULT_OK
|
|
|
|
|
&& (requestCode == REQUEST_CODE_OPEN_NODE || requestCode == REQUEST_CODE_NEW_NODE)) {
|
|
|
|
|
// 更新笔记列表适配器的数据源为null,触发UI重新拉取数据
|
|
|
|
|
mNotesListAdapter.changeCursor(null);
|
|
|
|
|
} else {
|
|
|
|
|
// 其他情况交由父类处理
|
|
|
|
|
super.onActivityResult(requestCode, resultCode, data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 从原始资源中设置应用信息
|
|
|
|
|
* 该方法主要用于首次添加应用时,从raw资源中读取介绍文本,并保存为工作便笺
|
|
|
|
|
*/
|
|
|
|
|
private void setAppInfoFromRawRes() {
|
|
|
|
|
// 获取SharedPreferences实例,用于存储应用的首选项
|
|
|
|
|
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this);
|
|
|
|
|
|
|
|
|
|
// 检查是否已经添加过介绍,如果已经添加,则不执行后续操作
|
|
|
|
|
if (!sp.getBoolean(PREFERENCE_ADD_INTRODUCTION, false)) {
|
|
|
|
|
// 使用StringBuilder来拼接读取的介绍文本
|
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
|
|
|
|
|
|
// 定义InputStream实例,用于读取raw资源
|
|
|
|
|
InputStream in = null;
|
|
|
|
|
try {
|
|
|
|
|
// 打开raw资源文件
|
|
|
|
|
in = getResources().openRawResource(R.raw.introduction);
|
|
|
|
|
in = getResources().openRawResource(R.raw.introduction);
|
|
|
|
|
if (in != null) {
|
|
|
|
|
// 将InputStream包装为InputStreamReader,以便按字符读取
|
|
|
|
|
InputStreamReader isr = new InputStreamReader(in);
|
|
|
|
|
// 再将InputStreamReader包装为BufferedReader,提高读取效率
|
|
|
|
|
BufferedReader br = new BufferedReader(isr);
|
|
|
|
|
char[] buf = new char[1024];
|
|
|
|
|
char [] buf = new char[1024];
|
|
|
|
|
int len = 0;
|
|
|
|
|
// 循环读取文件内容,直到读取结束
|
|
|
|
|
while ((len = br.read(buf)) > 0) {
|
|
|
|
|
sb.append(buf, 0, len);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// 如果无法打开raw资源文件,记录错误日志并退出方法
|
|
|
|
|
Log.e(TAG, "Read introduction file error");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} catch (IOException e) {
|
|
|
|
|
// 捕获IOException异常,记录堆栈信息并退出方法
|
|
|
|
|
e.printStackTrace();
|
|
|
|
|
return;
|
|
|
|
|
} finally {
|
|
|
|
|
// 确保关闭InputStream资源,避免资源泄露
|
|
|
|
|
if (in != null) {
|
|
|
|
|
if(in != null) {
|
|
|
|
|
try {
|
|
|
|
|
in.close();
|
|
|
|
|
} catch (IOException e) {
|
|
|
|
|
@ -211,17 +190,13 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 创建一个空的工作便笺,用于存储应用介绍文本
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
@ -261,78 +236,50 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
private ActionMode mActionMode;
|
|
|
|
|
private MenuItem mMoveMenu;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 创建操作模式并设置相关操作和视图
|
|
|
|
|
* 在此方法中,我们会根据当前的笔记状态设置不同的操作选项,并且根据用户的选择做出响应
|
|
|
|
|
* @param mode 当前的操作模式
|
|
|
|
|
* @param menu 菜单对象,用于充气和设置菜单项的可见性以及监听器
|
|
|
|
|
* @return 返回true表示处理了这个事件
|
|
|
|
|
*/
|
|
|
|
|
public boolean onCreateActionMode(ActionMode mode, Menu 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;
|
|
|
|
|
// 设置笔记列表适配器的选择模式为true,允许项目选择
|
|
|
|
|
mNotesListAdapter.setChoiceMode(true);
|
|
|
|
|
// 禁用长按可点击,防止弹出上下文菜单
|
|
|
|
|
mNotesListView.setLongClickable(false);
|
|
|
|
|
// 隐藏添加新笔记的按钮,因为当前在操作模式下
|
|
|
|
|
mAddNewNote.setVisibility(View.GONE);
|
|
|
|
|
|
|
|
|
|
// 创建自定义视图并充气布局,用于操作模式的自定义视图
|
|
|
|
|
View customView = LayoutInflater.from(NotesListActivity.this)
|
|
|
|
|
.inflate(R.layout.note_list_dropdown_menu, null);
|
|
|
|
|
// 设置操作模式的自定义视图为刚才创建的视图
|
|
|
|
|
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() {
|
|
|
|
|
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();
|
|
|
|
|
// 更新下拉菜单标题,使用资源文件中的格式,并填入已选择的笔记数量
|
|
|
|
|
// Update dropdown menu
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
@ -365,76 +312,53 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
updateMenu();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 处理菜单项点击事件
|
|
|
|
|
*
|
|
|
|
|
* @param item 被点击的菜单项
|
|
|
|
|
* @return 如果菜单项被处理,则返回true;否则返回false
|
|
|
|
|
*/
|
|
|
|
|
public boolean onMenuItemClick(MenuItem item) {
|
|
|
|
|
// 检查是否有选中的便签,如果没有选中任何便签,则提示用户并返回true
|
|
|
|
|
if (mNotesListAdapter.getSelectedCount() == 0) {
|
|
|
|
|
Toast.makeText(NotesListActivity.this, getString(R.string.menu_select_none),
|
|
|
|
|
Toast.LENGTH_SHORT).show();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 根据点击的菜单项ID进行不同的操作
|
|
|
|
|
switch (item.getItemId()) {
|
|
|
|
|
case R.id.delete:
|
|
|
|
|
// 创建确认删除对话框
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
builder.setTitle(getString(R.string.alert_title_delete));
|
|
|
|
|
builder.setIcon(android.R.drawable.ic_dialog_alert);
|
|
|
|
|
builder.setMessage(getString(R.string.alert_message_delete_notes,
|
|
|
|
|
mNotesListAdapter.getSelectedCount()));
|
|
|
|
|
mNotesListAdapter.getSelectedCount()));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok,
|
|
|
|
|
new DialogInterface.OnClickListener() {
|
|
|
|
|
public void onClick(DialogInterface dialog,
|
|
|
|
|
int which) {
|
|
|
|
|
batchDelete();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
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:
|
|
|
|
|
// 如果菜单项ID不匹配任何已知选项,则返回false
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// 如果处理了菜单项,则返回true
|
|
|
|
|
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();
|
|
|
|
|
// 如果状态为子文件夹编辑状态,需要减去标题栏的高度
|
|
|
|
|
/**
|
|
|
|
|
* Minus TitleBar's height
|
|
|
|
|
*/
|
|
|
|
|
if (mState == ListEditState.SUB_FOLDER) {
|
|
|
|
|
eventY -= mTitleBar.getHeight();
|
|
|
|
|
start -= mTitleBar.getHeight();
|
|
|
|
|
@ -448,65 +372,45 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
* 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))) {
|
|
|
|
|
// 初始化原始Y坐标和分发Y坐标
|
|
|
|
|
mOriginY = (int) event.getY();
|
|
|
|
|
mDispatchY = eventY;
|
|
|
|
|
// 设置事件位置为分发Y坐标
|
|
|
|
|
event.setLocation(event.getX(), mDispatchY);
|
|
|
|
|
// 标记为需要分发事件
|
|
|
|
|
mDispatch = true;
|
|
|
|
|
// 分发触摸事件到列表视图并返回结果
|
|
|
|
|
return mNotesListView.dispatchTouchEvent(event);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case MotionEvent.ACTION_MOVE: {
|
|
|
|
|
// 如果事件需要分发,则更新分发Y坐标并重新设置事件位置
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// 如果事件没有被处理,则返回false
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 异步启动笔记列表查询
|
|
|
|
|
* 根据当前文件夹ID配置查询条件,然后启动背景查询处理程序执行查询
|
|
|
|
|
* 查询笔记内容,排序依据笔记类型和修改日期降序排列
|
|
|
|
|
*/
|
|
|
|
|
private void startAsyncNotesListQuery() {
|
|
|
|
|
// 根据当前文件夹ID选择适当的查询条件
|
|
|
|
|
String selection = (mCurrentFolderId == Notes.ID_ROOT_FOLDER) ? ROOT_FOLDER_SELECTION
|
|
|
|
|
: NORMAL_SELECTION;
|
|
|
|
|
// 启动背景查询处理程序执行笔记列表查询
|
|
|
|
|
// 参数包括查询标识符、外部上下文、查询的URI、投影数组、选择条件和参数以及排序顺序
|
|
|
|
|
mBackgroundQueryHandler.startQuery(FOLDER_NOTE_LIST_QUERY_TOKEN, null,
|
|
|
|
|
Notes.CONTENT_NOTE_URI, NoteItemData.PROJECTION, selection, new String[] {
|
|
|
|
|
String.valueOf(mCurrentFolderId)
|
|
|
|
|
@ -518,77 +422,43 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
super(contentResolver);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 当数据库查询完成时调用的方法
|
|
|
|
|
*
|
|
|
|
|
* @param token 查询的标识符,用于区分不同的查询任务
|
|
|
|
|
* @param cookie 查询操作的附加数据,在此例程中未使用
|
|
|
|
|
* @param cursor 查询结果的游标对象,包含查询到的数据
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
protected void onQueryComplete(int token, Object cookie, Cursor cursor) {
|
|
|
|
|
// 根据不同的token值执行相应的处理
|
|
|
|
|
switch (token) {
|
|
|
|
|
// 当token为获取笔记列表的查询标识时
|
|
|
|
|
case FOLDER_NOTE_LIST_QUERY_TOKEN:
|
|
|
|
|
// 更新笔记列表的适配器,以显示新的查询结果
|
|
|
|
|
mNotesListAdapter.changeCursor(cursor);
|
|
|
|
|
break;
|
|
|
|
|
// 当token为获取文件夹列表的查询标识时
|
|
|
|
|
case FOLDER_LIST_QUERY_TOKEN:
|
|
|
|
|
// 检查查询结果是否有效且包含数据
|
|
|
|
|
if (cursor != null && cursor.getCount() > 0) {
|
|
|
|
|
// 显示文件夹列表菜单,传递查询结果作为参数
|
|
|
|
|
showFolderListMenu(cursor);
|
|
|
|
|
} else {
|
|
|
|
|
// 记录查询失败的日志信息
|
|
|
|
|
Log.e(TAG, "Query folder failed");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
// 对于其他token值,不做任何处理
|
|
|
|
|
default:
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 显示文件夹列表菜单
|
|
|
|
|
*
|
|
|
|
|
* @param cursor 游标对象,用于填充文件夹适配器的数据
|
|
|
|
|
*/
|
|
|
|
|
private void showFolderListMenu(Cursor cursor) {
|
|
|
|
|
// 创建一个对话框构建器
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
// 设置对话框标题为“选择文件夹”
|
|
|
|
|
builder.setTitle(R.string.menu_title_select_folder);
|
|
|
|
|
// 创建一个文件夹列表适配器,并使用传入的游标填充数据
|
|
|
|
|
final FoldersListAdapter adapter = new FoldersListAdapter(this, cursor);
|
|
|
|
|
// 设置适配器和对话框项点击监听器
|
|
|
|
|
builder.setAdapter(adapter, new DialogInterface.OnClickListener() {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 当对话框中的项被点击时调用
|
|
|
|
|
*
|
|
|
|
|
* @param dialog 被点击的对话框
|
|
|
|
|
* @param which 被点击的对话框项的索引
|
|
|
|
|
*/
|
|
|
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
|
|
|
// 批量移动选中的便签到点击的文件夹
|
|
|
|
|
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();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@ -599,51 +469,30 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
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 not synced, delete notes directly
|
|
|
|
|
if (DataUtils.batchDeleteNotes(mContentResolver, mNotesListAdapter
|
|
|
|
|
.getSelectedItemIds())) {
|
|
|
|
|
// 删除成功,不需额外处理
|
|
|
|
|
} else {
|
|
|
|
|
// 删除失败,记录错误日志
|
|
|
|
|
Log.e(TAG, "Delete notes error, should not happens");
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// 如果是同步模式,将删除的便签移动到回收站文件夹
|
|
|
|
|
// in sync mode, we'll move the deleted note into the trash
|
|
|
|
|
// folder
|
|
|
|
|
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) {
|
|
|
|
|
if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID
|
|
|
|
|
@ -652,44 +501,28 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// 结束操作模式
|
|
|
|
|
mModeCallBack.finishActionMode();
|
|
|
|
|
}
|
|
|
|
|
}.execute();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 删除指定的文件夹
|
|
|
|
|
* 如果文件夹是根文件夹,则记录错误并返回
|
|
|
|
|
* 在非同步模式下,直接删除文件夹;在同步模式下,将删除的文件夹移动到回收站
|
|
|
|
|
* 如果文件夹关联了小部件,则更新小部件
|
|
|
|
|
*
|
|
|
|
|
* @param folderId 要删除的文件夹的ID
|
|
|
|
|
*/
|
|
|
|
|
private void deleteFolder(long folderId) {
|
|
|
|
|
// 检查是否尝试删除根文件夹,记录错误并退出
|
|
|
|
|
if (folderId == Notes.ID_ROOT_FOLDER) {
|
|
|
|
|
Log.e(TAG, "Wrong folder id, should not happen " + folderId);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 使用HashSet存储待删除的文件夹ID
|
|
|
|
|
HashSet<Long> ids = new HashSet<Long>();
|
|
|
|
|
ids.add(folderId);
|
|
|
|
|
|
|
|
|
|
// 获取与文件夹关联的小部件集合
|
|
|
|
|
HashSet<AppWidgetAttribute> widgets = DataUtils.getFolderNoteWidget(mContentResolver, folderId);
|
|
|
|
|
|
|
|
|
|
// 根据同步模式选择删除方式
|
|
|
|
|
HashSet<AppWidgetAttribute> widgets = DataUtils.getFolderNoteWidget(mContentResolver,
|
|
|
|
|
folderId);
|
|
|
|
|
if (!isSyncMode()) {
|
|
|
|
|
// 如果不是同步模式,直接删除文件夹
|
|
|
|
|
// if not synced, delete folder directly
|
|
|
|
|
DataUtils.batchDeleteNotes(mContentResolver, ids);
|
|
|
|
|
} else {
|
|
|
|
|
// 如果是同步模式,将删除的文件夹移动到回收站
|
|
|
|
|
// in sync mode, we'll move the deleted folder into the trash folder
|
|
|
|
|
DataUtils.batchMoveToFolder(mContentResolver, ids, Notes.ID_TRASH_FOLER);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 如果存在关联的小部件,遍历并更新小部件
|
|
|
|
|
if (widgets != null) {
|
|
|
|
|
for (AppWidgetAttribute widget : widgets) {
|
|
|
|
|
if (widget.widgetId != AppWidgetManager.INVALID_APPWIDGET_ID
|
|
|
|
|
@ -707,36 +540,20 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
this.startActivityForResult(intent, REQUEST_CODE_OPEN_NODE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 打开指定的文件夹,并设置UI状态
|
|
|
|
|
*
|
|
|
|
|
* @param data 要打开的文件夹的NoteItemData对象,包含文件夹的相关信息
|
|
|
|
|
*/
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置标题栏文本,如果是通话记录文件夹,则使用资源文件中的名称
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@ -762,70 +579,43 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
inputMethodManager.hideSoftInputFromWindow(view.getWindowToken(), 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 显示创建或修改文件夹的对话框
|
|
|
|
|
*
|
|
|
|
|
* @param create 如果为true,则显示创建文件夹的对话框;如果为false,则显示修改文件夹名称的对话框
|
|
|
|
|
*/
|
|
|
|
|
private void showCreateOrModifyFolderDialog(final boolean create) {
|
|
|
|
|
// 使用AlertDialog.Builder创建对话框
|
|
|
|
|
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();
|
|
|
|
|
@ -837,23 +627,22 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
// 监听编辑文本的变化,以启用或禁用“确定”按钮
|
|
|
|
|
/**
|
|
|
|
|
* 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
|
|
|
|
|
@ -861,7 +650,6 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void onTextChanged(CharSequence s, int start, int before, int count) {
|
|
|
|
|
// 根据输入文本的长度来启用或禁用“确定”按钮
|
|
|
|
|
if (TextUtils.isEmpty(etName.getText())) {
|
|
|
|
|
positive.setEnabled(false);
|
|
|
|
|
} else {
|
|
|
|
|
@ -877,72 +665,45 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
/**
|
|
|
|
|
* 处理后退按钮按下事件
|
|
|
|
|
* 根据当前的状态不同,执行不同的后退逻辑
|
|
|
|
|
*/
|
|
|
|
|
public void onBackPressed() {
|
|
|
|
|
// 根据当前状态选择相应的后退行为
|
|
|
|
|
switch (mState) {
|
|
|
|
|
case SUB_FOLDER:
|
|
|
|
|
// 当前在子文件夹中,返回上级文件夹
|
|
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER; // 设置当前文件夹ID为根文件夹ID
|
|
|
|
|
mState = ListEditState.NOTE_LIST; // 更改状态为笔记列表
|
|
|
|
|
startAsyncNotesListQuery(); // 开始异步查询笔记列表
|
|
|
|
|
mTitleBar.setVisibility(View.GONE); // 隐藏标题栏
|
|
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER;
|
|
|
|
|
mState = ListEditState.NOTE_LIST;
|
|
|
|
|
startAsyncNotesListQuery();
|
|
|
|
|
mTitleBar.setVisibility(View.GONE);
|
|
|
|
|
break;
|
|
|
|
|
case CALL_RECORD_FOLDER:
|
|
|
|
|
// 当前在通话录音文件夹中,返回笔记列表
|
|
|
|
|
mCurrentFolderId = Notes.ID_ROOT_FOLDER; // 设置当前文件夹ID为根文件夹ID
|
|
|
|
|
mState = ListEditState.NOTE_LIST; // 更改状态为笔记列表
|
|
|
|
|
mAddNewNote.setVisibility(View.VISIBLE); // 显示添加新笔记按钮
|
|
|
|
|
mTitleBar.setVisibility(View.GONE); // 隐藏标题栏
|
|
|
|
|
startAsyncNotesListQuery(); // 开始异步查询笔记列表
|
|
|
|
|
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 小部件的唯一标识符
|
|
|
|
|
* @param appWidgetType 小部件的类型,决定如何更新
|
|
|
|
|
*
|
|
|
|
|
* 本方法通过发送广播来请求更新特定类型的小部件它首先根据小部件类型选择正确的
|
|
|
|
|
* 小部件提供者类,然后构造一个意图并指定小部件的ID,最后广播这个意图以触发小部件
|
|
|
|
|
* 的更新过程如果指定的类型不受支持,则记录错误并终止方法执行
|
|
|
|
|
*/
|
|
|
|
|
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);
|
|
|
|
|
} else if (appWidgetType == Notes.TYPE_WIDGET_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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@ -967,19 +728,15 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
|
|
|
|
|
@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);
|
|
|
|
|
@ -987,7 +744,6 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok,
|
|
|
|
|
new DialogInterface.OnClickListener() {
|
|
|
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
|
|
|
// 确认删除当前焦点的数据项对应的文件夹
|
|
|
|
|
deleteFolder(mFocusNoteDataItem.getId());
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
@ -995,94 +751,67 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
builder.show();
|
|
|
|
|
break;
|
|
|
|
|
case MENU_FOLDER_CHANGE_NAME:
|
|
|
|
|
// 显示修改文件夹名称的对话框
|
|
|
|
|
showCreateOrModifyFolderDialog(false);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
// 其他情况不做操作
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 在选项菜单显示前进行准备。
|
|
|
|
|
* 根据当前状态设置不同的菜单项。
|
|
|
|
|
*
|
|
|
|
|
* @param menu 菜单对象
|
|
|
|
|
* @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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@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: {
|
|
|
|
|
// 处理搜索请求
|
|
|
|
|
case R.id.menu_search:
|
|
|
|
|
onSearchRequested();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
@ -1095,35 +824,17 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 将便签导出为文本文件
|
|
|
|
|
* 该方法使用BackupUtils类的功能将便签数据导出为文本文件
|
|
|
|
|
* 导出操作在后台线程中执行,以避免阻塞主线程
|
|
|
|
|
*/
|
|
|
|
|
private void exportNoteToText() {
|
|
|
|
|
// 获取BackupUtils的实例,用于执行备份操作
|
|
|
|
|
final BackupUtils backup = BackupUtils.getInstance(NotesListActivity.this);
|
|
|
|
|
// 使用AsyncTask在后台执行导出操作
|
|
|
|
|
new AsyncTask<Void, Void, Integer>() {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 在后台线程中执行的耗时操作
|
|
|
|
|
* 此方法调用BackupUtils的exportToText方法进行文本文件的导出
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
protected Integer doInBackground(Void... unused) {
|
|
|
|
|
return backup.exportToText();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 在后台操作完成后执行的操作
|
|
|
|
|
* 根据导出结果展示不同的对话框通知用户
|
|
|
|
|
*
|
|
|
|
|
* @param result 导出操作的结果,可能的值包括STATE_SD_CARD_UNMOUONTED、STATE_SUCCESS、STATE_SYSTEM_ERROR
|
|
|
|
|
*/
|
|
|
|
|
@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
|
|
|
|
|
@ -1132,8 +843,7 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
.getString(R.string.error_sdcard_unmounted));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok, null);
|
|
|
|
|
builder.show();
|
|
|
|
|
} // 导出成功时的处理
|
|
|
|
|
else if (result == BackupUtils.STATE_SUCCESS) {
|
|
|
|
|
} else if (result == BackupUtils.STATE_SUCCESS) {
|
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(NotesListActivity.this);
|
|
|
|
|
builder.setTitle(NotesListActivity.this
|
|
|
|
|
.getString(R.string.success_sdcard_export));
|
|
|
|
|
@ -1142,8 +852,7 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
.getExportedTextFileName(), backup.getExportedTextFileDir()));
|
|
|
|
|
builder.setPositiveButton(android.R.string.ok, null);
|
|
|
|
|
builder.show();
|
|
|
|
|
} // 系统错误时的处理
|
|
|
|
|
else if (result == BackupUtils.STATE_SYSTEM_ERROR) {
|
|
|
|
|
} 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));
|
|
|
|
|
@ -1208,20 +917,11 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 启动查询目标文件夹的操作
|
|
|
|
|
*
|
|
|
|
|
* 此方法用于构建查询条件并启动一个查询任务,该任务将根据当前的应用状态或编辑状态
|
|
|
|
|
* 查询相关的文件夹列表。查询结果将用于展示或编辑笔记。
|
|
|
|
|
*/
|
|
|
|
|
private void startQueryDestinationFolders() {
|
|
|
|
|
// 定义查询条件的SQL语句模板
|
|
|
|
|
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,
|
|
|
|
|
@ -1235,41 +935,20 @@ public class NotesListActivity extends Activity implements OnClickListener, OnIt
|
|
|
|
|
NoteColumns.MODIFIED_DATE + " DESC");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 处理列表项长按事件的方法
|
|
|
|
|
*
|
|
|
|
|
* 当用户在列表项上长按时,此方法会被调用它根据当前焦点的笔记类型(单个笔记或文件夹)
|
|
|
|
|
* 来决定是启动选择模式还是准备处理文件夹的上下文菜单
|
|
|
|
|
*
|
|
|
|
|
* @param parent 触发事件的父适配器
|
|
|
|
|
* @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()) {
|
|
|
|
|
// 尝试启动选择模式
|
|
|
|
|
if (mNotesListView.startActionMode(mModeCallBack) != null) {
|
|
|
|
|
// 选择当前项并给予触觉反馈
|
|
|
|
|
mModeCallBack.onItemCheckedStateChanged(null, position, id, true);
|
|
|
|
|
mNotesListView.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
|
|
|
|
|
} else {
|
|
|
|
|
// 如果启动选择模式失败,则记录错误日志
|
|
|
|
|
Log.e(TAG, "startActionMode fails");
|
|
|
|
|
}
|
|
|
|
|
} else if (mFocusNoteDataItem.getType() == Notes.TYPE_FOLDER) {
|
|
|
|
|
// 如果是文件夹,则设置文件夹的上下文菜单监听器
|
|
|
|
|
mNotesListView.setOnCreateContextMenuListener(mFolderOnCreateContextMenuListener);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// 不拦截事件,允许其他监听器处理
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|