You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
GPL_Project/doc/标注/210340061_林亭旭/TaskList.java

436 lines
19 KiB

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

/*
* 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.gtask.data;
import android.database.Cursor;
import android.util.Log;
import net.micode.notes.data.Notes;
import net.micode.notes.data.Notes.NoteColumns;
import net.micode.notes.gtask.exception.ActionFailureException;
import net.micode.notes.tool.GTaskStringUtils;
import org.json.JSONException;
import org.json.JSONObject;
import java.util.ArrayList;
public class TaskList extends Node
//TaskList 类继承了 Node 类
// 表示它是可以存储在树形结构中的一个节点
// TaskList 类有三个属性mIndex、mChildren 和 TAG
{
private static final String TAG = TaskList.class.getSimpleName();
//用于记录 TaskList 类的类名,方便在调试过程中进行日志记录和错误跟踪
private int mIndex;
//表示 TaskList 对象在树形结构中的序号。默认情况下,它被初始化为 1
private ArrayList<Task> mChildren;
// ArrayList<Task> 类型的属性,表示 TaskList 对象的子节点列表
public TaskList() {
super();
mChildren = new ArrayList<Task>();
mIndex = 1;
}
public JSONObject getCreateAction(int actionId)
//据当前文件夹对象的属性值,构造一个包含创建操作信息的 JSON 对象,并返回给调用者
{
JSONObject js = new JSONObject();
//创建一个空的 JSONObject 对象 js
try {
// action_type
js.put(GTaskStringUtils.GTASK_JSON_ACTION_TYPE,
GTaskStringUtils.GTASK_JSON_ACTION_TYPE_CREATE);
//添加 GTASK_JSON_ACTION_TYPE字段及对应的值
// action_id
js.put(GTaskStringUtils.GTASK_JSON_ACTION_ID, actionId);
// 添加GTASK_JSON_ACTION_ID字段及对应的值
// index
js.put(GTaskStringUtils.GTASK_JSON_INDEX, mIndex);
// entity_delta
// 添加GTASK_JSON_INDEX 和 GTASK_JSON_ENTITY_DELTA 等字段及对应的值。
JSONObject entity = new JSONObject();
entity.put(GTaskStringUtils.GTASK_JSON_NAME, getName());
//创建一个包含实体信息的 JSONObject 对象 entity并向其中添加文件夹名称 GTASK_JSON_NAME
entity.put(GTaskStringUtils.GTASK_JSON_CREATOR_ID, "null");
entity.put(GTaskStringUtils.GTASK_JSON_ENTITY_TYPE,
GTaskStringUtils.GTASK_JSON_TYPE_GROUP);
//创建人 ID GTASK_JSON_CREATOR_ID 和实体类型 GTASK_JSON_ENTITY_TYPE 等字段及对应的值
js.put(GTaskStringUtils.GTASK_JSON_ENTITY_DELTA, entity);
//将 entity 对象作为值添加到父级 JSONObject 对象 js 中的 GTASK_JSON_ENTITY_DELTA 字段中
} catch (JSONException e) {
Log.e(TAG, e.toString());
e.printStackTrace();
throw new ActionFailureException("fail to generate tasklist-create jsonobject");
}//如果生成 JSON 对象时出现异常,则记录错误日志并抛出 ActionFailureException 异常
return js;
}
public JSONObject getUpdateAction(int actionId)
//根据当前文件夹对象的属性值,构造一个包含更新操作信息的 JSON 对象,并返回给调用者
{
JSONObject js = new JSONObject();//创建一个空的 JSONObject 对象 js
try {
// action_type
js.put(GTaskStringUtils.GTASK_JSON_ACTION_TYPE,
GTaskStringUtils.GTASK_JSON_ACTION_TYPE_UPDATE);
//添加 GTASK_JSON_ACTION_TYPE字段及对应的值
// action_id
js.put(GTaskStringUtils.GTASK_JSON_ACTION_ID, actionId);
//添加GTASK_JSON_ACTION_ID字段及对应的值
// id
js.put(GTaskStringUtils.GTASK_JSON_ID, getGid());
//添加GTASK_JSON_ID字段及对应的值
// entity_delta
JSONObject entity = new JSONObject();
entity.put(GTaskStringUtils.GTASK_JSON_NAME, getName());
entity.put(GTaskStringUtils.GTASK_JSON_DELETED, getDeleted());
//添加GTASK_JSON_DELETED字段及对应的值
js.put(GTaskStringUtils.GTASK_JSON_ENTITY_DELTA, entity);
////然后将 entity 对象作为值添加到父级 JSONObject 对象 js 中的 GTASK_JSON_ENTITY_DELTA 字段
} catch (JSONException e) {
Log.e(TAG, e.toString());
e.printStackTrace();
throw new ActionFailureException("fail to generate tasklist-update jsonobject");
}
//如果生成 JSON 对象时出现异常,则记录错误日志并抛出 ActionFailureException 异常
return js;
}
public void setContentByRemoteJSON(JSONObject js)
//从给定的 JSONObject 对象 js 中读取云端 Google Tasks 的文件夹信息,并将其应用到当前的文件夹对象中
{
if (js != null)
//对输入进行判空
{
try {
// id
if (js.has(GTaskStringUtils.GTASK_JSON_ID))
//检查是否含有字段 GTASK_JSON_ID
{
setGid(js.getString(GTaskStringUtils.GTASK_JSON_ID));
}
// last_modified
if (js.has(GTaskStringUtils.GTASK_JSON_LAST_MODIFIED)) {
setLastModified(js.getLong(GTaskStringUtils.GTASK_JSON_LAST_MODIFIED));
}//检查是否含有字段 GTASK_JSON_LAST_MODIFIED
// name
if (js.has(GTaskStringUtils.GTASK_JSON_NAME)) {
setName(js.getString(GTaskStringUtils.GTASK_JSON_NAME));
}//检查是否含有字段GTASK_JSON_NAME
} catch (JSONException e) {
Log.e(TAG, e.toString());
e.printStackTrace();
throw new ActionFailureException("fail to get tasklist content from jsonobject");
}
//如果解析 JSON 对象时出现异常
// 则记录错误日志并抛出 ActionFailureException 异常
}
}
public void setContentByLocalJSON(JSONObject js)
//从给定的 JSONObject 对象 js 中读取文件夹信息,并将其应用到当前的文件夹对象中
{
if (js == null || !js.has(GTaskStringUtils.META_HEAD_NOTE)) {
Log.w(TAG, "setContentByLocalJSON: nothing is avaiable");
}
//如果输入的 JSON 对象为 null 或不含有 GTaskStringUtils.META_HEAD_NOTE 字段
// 则会记录警告日志并跳过后续处理
try {
JSONObject folder = js.getJSONObject(GTaskStringUtils.META_HEAD_NOTE);
//从 GTaskStringUtils.META_HEAD_NOTE 字段中获取 folder 子对象,并根据 NoteColumns.TYPE 属性判断文件夹类型
if (folder.getInt(NoteColumns.TYPE) == Notes.TYPE_FOLDER)
// 如果文件夹类型是 Notes.TYPE_FOLDER
{
String name = folder.getString(NoteColumns.SNIPPET);//则从 NoteColumns.SNIPPET 属性中获取文件夹名称
setName(GTaskStringUtils.MIUI_FOLDER_PREFFIX + name);//为当前文件夹对象设置名称,以 GTaskStringUtils.MIUI_FOLDER_PREFFIX 作为前缀
} else if (folder.getInt(NoteColumns.TYPE) == Notes.TYPE_SYSTEM)
//如果文件夹类型是 Notes.TYPE_SYSTEM
{
if (folder.getLong(NoteColumns.ID) == Notes.ID_ROOT_FOLDER)
//根据 NoteColumns.ID 属性的值判断系统文件夹类型,并为当前文件夹对象设置相应的名称
setName(GTaskStringUtils.MIUI_FOLDER_PREFFIX + GTaskStringUtils.FOLDER_DEFAULT);
else if (folder.getLong(NoteColumns.ID) == Notes.ID_CALL_RECORD_FOLDER)
setName(GTaskStringUtils.MIUI_FOLDER_PREFFIX
+ GTaskStringUtils.FOLDER_CALL_NOTE);
else
Log.e(TAG, "invalid system folder");
} else {
Log.e(TAG, "error type");
}//如果出现错误,则记录错误日志并跳过后续处理。如果文件夹类型不合法,则同样记录错误日志并跳过后续处理。
} catch (JSONException e) {
Log.e(TAG, e.toString());
e.printStackTrace();
}
}
public JSONObject getLocalJSONFromContent()
//将应用中的文件夹信息转换为 JSON 格式的方法
//返回一个包含文件夹信息的 JSONObject 对象
{
try {
JSONObject js = new JSONObject();
JSONObject folder = new JSONObject();
//创建一个新的 JSONObject 对象
// 并创建一个名为 folder 的子对象,用于存储文件夹的属性信息
String folderName = getName();
if (getName().startsWith(GTaskStringUtils.MIUI_FOLDER_PREFFIX))
folderName = folderName.substring(GTaskStringUtils.MIUI_FOLDER_PREFFIX.length(),
folderName.length());
//从当前文件夹对象中获取文件夹名称,并根据名称的前缀判断该文件夹是否是系统文件夹
folder.put(NoteColumns.SNIPPET, folderName);
if (folderName.equals(GTaskStringUtils.FOLDER_DEFAULT)
|| folderName.equals(GTaskStringUtils.FOLDER_CALL_NOTE))
folder.put(NoteColumns.TYPE, Notes.TYPE_SYSTEM);
// //如果是系统文件夹,则设置 NoteColumns.TYPE 属性为 Notes.TYPE_SYSTEM
else
folder.put(NoteColumns.TYPE, Notes.TYPE_FOLDER);//否则设置为 Notes.TYPE_FOLDER
js.put(GTaskStringUtils.META_HEAD_NOTE, folder);
//将文件夹的名称和类型信息添加到 folder 子对象中,并将 folder 作为 GTaskStringUtils.META_HEAD_NOTE 的值添加到主 JSONObject 中
// 用于指定待更新笔记的文件夹信息。
return js;
} catch (JSONException e) {
Log.e(TAG, e.toString());
e.printStackTrace();
return null;
}//如果转换过程中出现异常,则记录错误日志并返回 null
}
public int getSyncAction(Cursor c)
//从给定的 Cursor 对象 c 中读取本地笔记的信息,并将其与云端笔记进行比较,
// 从而决定应该执行哪种同步操作
{
try {
if (c.getInt(SqlNote.LOCAL_MODIFIED_COLUMN) == 0)
//判断本地笔记是否有修改
{
// there is no local update
if (c.getLong(SqlNote.SYNC_ID_COLUMN) == getLastModified()) {
// no update both side
//如果本地笔记没有被修改
// 则进一步判断云端笔记的最后修改时间和本地笔记最后同步时间是否相等
// 如果相等,则不需要更新,返回 SYNC_ACTION_NON
return SYNC_ACTION_NONE;
} else {
// apply remote to local
//如果不相等,则需要将云端笔记的内容同步到本地,返回 SYNC_ACTION_UPDATE_LOCAL
return SYNC_ACTION_UPDATE_LOCAL;
}
} else
//被修改过
{
// validate gtask id
if (!c.getString(SqlNote.GTASK_ID_COLUMN).equals(getGid()))
//验证云端笔记的 ID 和本地笔记的 ID 是否匹配
{
Log.e(TAG, "gtask id doesn't match");
return SYNC_ACTION_ERROR;
}//如果不匹配,则说明出现了错误
// 返回 SYNC_ACTION_ERROR 表示同步失败
if (c.getLong(SqlNote.SYNC_ID_COLUMN) == getLastModified()) {
// local modification only
return SYNC_ACTION_UPDATE_REMOTE;
//如果 ID 匹配,则需要进一步判断本地笔记和云端笔记的最后修改时间。
// 如果相等,则只需要将本地笔记同步到云端
//返回 SYNC_ACTION_UPDATE_REMOTE
} else {
// for folder conflicts, just apply local modification
return SYNC_ACTION_UPDATE_REMOTE;
}//存在冲突,优先选择本地笔记的修改内容
// 返回 SYNC_ACTION_UPDATE_REMOTE
}
} catch (Exception e) {
Log.e(TAG, e.toString());
e.printStackTrace();
}
return SYNC_ACTION_ERROR;
}
public int getChildTaskCount() {
return mChildren.size();
}
public boolean addChildTask(Task task)
//向任务列表中添加指定的子任务
{
boolean ret = false;
if (task != null && !mChildren.contains(task))
//判断待添加的任务是否为空,并且是否已经存在于列表中
//如果待添加任务为空,或者已经在列表中,则直接返回添加失败
{
ret = mChildren.add(task);
if (ret) {
// need to set prior sibling and parent
task.setPriorSibling(mChildren.isEmpty() ? null : mChildren
.get(mChildren.size() - 1));
task.setParent(this);//将待添加任务的前驱节点设置为列表中的最后一个任务(如果列表不为空,将其父节点设置为该任务列表的实例对象。
}//该方法返回操作结果,如果添加成功则返回 true否则返回 false。
}
return ret;
}
public boolean addChildTask(Task task, int index)
//用于向任务列表中添加指定的子任务
{
if (index < 0 || index > mChildren.size())
//判断待添加任务的索引是否越界
{
Log.e(TAG, "add child task: invalid index");
return false;
}//越界则记录一条错误日志,并返回 false 表示添加失败
int pos = mChildren.indexOf(task);
if (task != null && pos == -1) {
mChildren.add(index, task);
// update the task list
Task preTask = null;
Task afterTask = null;
if (index != 0)//如果待添加任务不在列表中,则将其插入到指定位置上,并更新列表中其他任务的“父兄弟”关系
preTask = mChildren.get(index - 1);
//将待添加任务的前驱节点设置为索引 index - 1
if (index != mChildren.size() - 1)
afterTask = mChildren.get(index + 1);
//后继节点设置为索引 index + 1 的任务
task.setPriorSibling(preTask);
if (afterTask != null)
afterTask.setPriorSibling(task);
}
return true;
}//待添加任务已经在列表中,则不执行操作并直接返回 true 表示添加成功
public boolean removeChildTask(Task task)
//从任务列表中删除指定的子任务
{
boolean ret = false;
int index = mChildren.indexOf(task);
if (index != -1) {
ret = mChildren.remove(task);
//通过 indexOf() 方法查找待删除任务在列表中的索引
if (ret) {
// reset prior sibling and parent
task.setPriorSibling(null);
task.setParent(null);
//执行删除操作,重置为 null
// update the task list
if (index != mChildren.size()) {
mChildren.get(index).setPriorSibling(
index == 0 ? null : mChildren.get(index - 1));
}
//更新列表中其他任务的关系:如果待删除任务的前一个任务存在,则将其设置为待删除任务的前驱节点;
// 否则,设置为 null。
}
}
return ret;
}
public boolean moveChildTask(Task task, int index)
//将指定的子任务移动到新的位置
{
if (index < 0 || index >= mChildren.size())//判断待移动任务的索引是否越界
{
Log.e(TAG, "move child task: invalid index");
return false;
}
//若存在越界,返回 false 表示操作失败
int pos = mChildren.indexOf(task);
if (pos == -1) {
Log.e(TAG, "move child task: the task should in the list");
return false;
}//如果待移动任务不在列表中,则会记录一条错误日志,并返回 false 表示操作失败
if (pos == index)////如果待移动任务已经在列表中,并且新的索引与当前位置相同
return true;
return (removeChildTask(task) && addChildTask(task, index));
}//直接返回 true 表示操作成功
public Task findChildTaskByGid(String gid)
//根据子任务的GID全局唯一标识符查找指定的子任务
{
for (int i = 0; i < mChildren.size(); i++)
//当前任务列表节点下所有的子任务中依次查找判断每个子任务的GID是否等于输入的参数gid
{
Task t = mChildren.get(i);
if (t.getGid().equals(gid)) {
return t;
}//若找到对应的子任务,则返回该子任务;
}
return null;
}//否则返回null表示找不到指定的子任务
public int getChildTaskIndex(Task task) {
return mChildren.indexOf(task);
}
public Task getChildTaskByIndex(int index)
//根据子任务在任务列表中的位置获取指定的子任务
{
if (index < 0 || index >= mChildren.size())
{
Log.e(TAG, "getTaskByIndex: invalid index");
return null;
}//判断输入的参数index是否越界如果小于0或大于等于当前子任务的个数则会记录一条错误日志并返回null
return mChildren.get(index);//返回该位置上对应的子任务
}
public Task getChilTaskByGid(String gid)
//根据子任务的GID全局唯一标识符查找指定的子任务
{
for (Task task : mChildren)
//遍历当前任务列表节点下所有的子任务并依次判断每个子任务的GID是否等于输入的参数gid
{
if (task.getGid().equals(gid))
return task;
}
return null;
}
//若找到对应的子任务则返回该子任务否则返回null表示找不到指定的子任务
public ArrayList<Task> getChildTaskList() {
return this.mChildren;
}
public void setIndex(int index) {
this.mIndex = index;
}
public int getIndex() {
return this.mIndex;
}
}