Task & TaksList & ActionFailureException类的注释 #17
Open
pp6wf43ns
wants to merge 0 commits from hy_branch into master
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -0,0 +1,253 @@
|
|||||||
|
/*
|
||||||
|
* 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.model;
|
||||||
|
import android.content.ContentProviderOperation;
|
||||||
|
import android.content.ContentProviderResult;
|
||||||
|
import android.content.ContentUris;
|
||||||
|
import android.content.ContentValues;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.OperationApplicationException;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.RemoteException;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import net.micode.notes.data.Notes;
|
||||||
|
import net.micode.notes.data.Notes.CallNote;
|
||||||
|
import net.micode.notes.data.Notes.DataColumns;
|
||||||
|
import net.micode.notes.data.Notes.NoteColumns;
|
||||||
|
import net.micode.notes.data.Notes.TextNote;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
|
|
||||||
|
public class Note {
|
||||||
|
private ContentValues mNoteDiffValues;
|
||||||
|
private NoteData mNoteData;
|
||||||
|
private static final String TAG = "Note";
|
||||||
|
/**
|
||||||
|
* Create a new note id for adding a new note to databases
|
||||||
|
*/
|
||||||
|
public static synchronized long getNewNoteId(Context context, long folderId) {
|
||||||
|
// Create a new note in the database
|
||||||
|
ContentValues values = new ContentValues();
|
||||||
|
long createdTime = System.currentTimeMillis();
|
||||||
|
values.put(NoteColumns.CREATED_DATE, createdTime);
|
||||||
|
values.put(NoteColumns.MODIFIED_DATE, createdTime);
|
||||||
|
values.put(NoteColumns.TYPE, Notes.TYPE_NOTE);
|
||||||
|
values.put(NoteColumns.LOCAL_MODIFIED, 1);
|
||||||
|
values.put(NoteColumns.PARENT_ID, folderId);
|
||||||
|
Uri uri = context.getContentResolver().insert(Notes.CONTENT_NOTE_URI, values);
|
||||||
|
|
||||||
|
long noteId = 0;
|
||||||
|
try {
|
||||||
|
noteId = Long.valueOf(uri.getPathSegments().get(1));
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
Log.e(TAG, "Get note id error :" + e.toString());
|
||||||
|
noteId = 0;
|
||||||
|
}
|
||||||
|
if (noteId == -1) {
|
||||||
|
throw new IllegalStateException("Wrong note id:" + noteId);
|
||||||
|
}
|
||||||
|
return noteId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Note() {
|
||||||
|
mNoteDiffValues = new ContentValues();
|
||||||
|
mNoteData = new NoteData();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setNoteValue(String key, String value) {
|
||||||
|
mNoteDiffValues.put(key, value);
|
||||||
|
mNoteDiffValues.put(NoteColumns.LOCAL_MODIFIED, 1);
|
||||||
|
mNoteDiffValues.put(NoteColumns.MODIFIED_DATE, System.currentTimeMillis());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTextData(String key, String value) {
|
||||||
|
mNoteData.setTextData(key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTextDataId(long id) {
|
||||||
|
mNoteData.setTextDataId(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTextDataId() {
|
||||||
|
return mNoteData.mTextDataId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCallDataId(long id) {
|
||||||
|
mNoteData.setCallDataId(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCallData(String key, String value) {
|
||||||
|
mNoteData.setCallData(key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isLocalModified() {
|
||||||
|
return mNoteDiffValues.size() > 0 || mNoteData.isLocalModified();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean syncNote(Context context, long noteId) {
|
||||||
|
if (noteId <= 0) {
|
||||||
|
throw new IllegalArgumentException("Wrong note id:" + noteId);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isLocalModified()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* In theory, once data changed, the note should be updated on {@link NoteColumns#LOCAL_MODIFIED} and
|
||||||
|
* {@link NoteColumns#MODIFIED_DATE}. For data safety, though update note fails, we also update the
|
||||||
|
* note data info
|
||||||
|
*/
|
||||||
|
if (context.getContentResolver().update(
|
||||||
|
ContentUris.withAppendedId(Notes.CONTENT_NOTE_URI, noteId), mNoteDiffValues, null,
|
||||||
|
null) == 0) {
|
||||||
|
Log.e(TAG, "Update note error, should not happen");
|
||||||
|
// Do not return, fall through
|
||||||
|
}
|
||||||
|
mNoteDiffValues.clear();
|
||||||
|
|
||||||
|
if (mNoteData.isLocalModified()
|
||||||
|
&& (mNoteData.pushIntoContentResolver(context, noteId) == null)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private class NoteData {
|
||||||
|
private long mTextDataId;
|
||||||
|
|
||||||
|
private ContentValues mTextDataValues;
|
||||||
|
|
||||||
|
private long mCallDataId;
|
||||||
|
|
||||||
|
private ContentValues mCallDataValues;
|
||||||
|
|
||||||
|
private static final String TAG = "NoteData";
|
||||||
|
|
||||||
|
public NoteData() {
|
||||||
|
mTextDataValues = new ContentValues();
|
||||||
|
mCallDataValues = new ContentValues();
|
||||||
|
mTextDataId = 0;
|
||||||
|
mCallDataId = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean isLocalModified() {
|
||||||
|
return mTextDataValues.size() > 0 || mCallDataValues.size() > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setTextDataId(long id) {
|
||||||
|
if(id <= 0) {
|
||||||
|
throw new IllegalArgumentException("Text data id should larger than 0");
|
||||||
|
}
|
||||||
|
mTextDataId = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setCallDataId(long id) {
|
||||||
|
if (id <= 0) {
|
||||||
|
throw new IllegalArgumentException("Call data id should larger than 0");
|
||||||
|
}
|
||||||
|
mCallDataId = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setCallData(String key, String value) {
|
||||||
|
mCallDataValues.put(key, value);
|
||||||
|
mNoteDiffValues.put(NoteColumns.LOCAL_MODIFIED, 1);
|
||||||
|
mNoteDiffValues.put(NoteColumns.MODIFIED_DATE, System.currentTimeMillis());
|
||||||
|
}
|
||||||
|
|
||||||
|
void setTextData(String key, String value) {
|
||||||
|
mTextDataValues.put(key, value);
|
||||||
|
mNoteDiffValues.put(NoteColumns.LOCAL_MODIFIED, 1);
|
||||||
|
mNoteDiffValues.put(NoteColumns.MODIFIED_DATE, System.currentTimeMillis());
|
||||||
|
}
|
||||||
|
|
||||||
|
Uri pushIntoContentResolver(Context context, long noteId) {
|
||||||
|
/**
|
||||||
|
* Check for safety
|
||||||
|
*/
|
||||||
|
if (noteId <= 0) {
|
||||||
|
throw new IllegalArgumentException("Wrong note id:" + noteId);
|
||||||
|
}
|
||||||
|
|
||||||
|
ArrayList<ContentProviderOperation> operationList = new ArrayList<ContentProviderOperation>();
|
||||||
|
ContentProviderOperation.Builder builder = null;
|
||||||
|
|
||||||
|
if(mTextDataValues.size() > 0) {
|
||||||
|
mTextDataValues.put(DataColumns.NOTE_ID, noteId);
|
||||||
|
if (mTextDataId == 0) {
|
||||||
|
mTextDataValues.put(DataColumns.MIME_TYPE, TextNote.CONTENT_ITEM_TYPE);
|
||||||
|
Uri uri = context.getContentResolver().insert(Notes.CONTENT_DATA_URI,
|
||||||
|
mTextDataValues);
|
||||||
|
try {
|
||||||
|
setTextDataId(Long.valueOf(uri.getPathSegments().get(1)));
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
Log.e(TAG, "Insert new text data fail with noteId" + noteId);
|
||||||
|
mTextDataValues.clear();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
builder = ContentProviderOperation.newUpdate(ContentUris.withAppendedId(
|
||||||
|
Notes.CONTENT_DATA_URI, mTextDataId));
|
||||||
|
builder.withValues(mTextDataValues);
|
||||||
|
operationList.add(builder.build());
|
||||||
|
}
|
||||||
|
mTextDataValues.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
if(mCallDataValues.size() > 0) {
|
||||||
|
mCallDataValues.put(DataColumns.NOTE_ID, noteId);
|
||||||
|
if (mCallDataId == 0) {
|
||||||
|
mCallDataValues.put(DataColumns.MIME_TYPE, CallNote.CONTENT_ITEM_TYPE);
|
||||||
|
Uri uri = context.getContentResolver().insert(Notes.CONTENT_DATA_URI,
|
||||||
|
mCallDataValues);
|
||||||
|
try {
|
||||||
|
setCallDataId(Long.valueOf(uri.getPathSegments().get(1)));
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
Log.e(TAG, "Insert new call data fail with noteId" + noteId);
|
||||||
|
mCallDataValues.clear();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
builder = ContentProviderOperation.newUpdate(ContentUris.withAppendedId(
|
||||||
|
Notes.CONTENT_DATA_URI, mCallDataId));
|
||||||
|
builder.withValues(mCallDataValues);
|
||||||
|
operationList.add(builder.build());
|
||||||
|
}
|
||||||
|
mCallDataValues.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (operationList.size() > 0) {
|
||||||
|
try {
|
||||||
|
ContentProviderResult[] results = context.getContentResolver().applyBatch(
|
||||||
|
Notes.AUTHORITY, operationList);
|
||||||
|
return (results == null || results.length == 0 || results[0] == null) ? null
|
||||||
|
: ContentUris.withAppendedId(Notes.CONTENT_NOTE_URI, noteId);
|
||||||
|
} catch (RemoteException e) {
|
||||||
|
Log.e(TAG, String.format("%s: %s", e.toString(), e.getMessage()));
|
||||||
|
return null;
|
||||||
|
} catch (OperationApplicationException e) {
|
||||||
|
Log.e(TAG, String.format("%s: %s", e.toString(), e.getMessage()));
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,368 @@
|
|||||||
|
/*
|
||||||
|
* 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.model;
|
||||||
|
|
||||||
|
import android.appwidget.AppWidgetManager;
|
||||||
|
import android.content.ContentUris;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.database.Cursor;
|
||||||
|
import android.text.TextUtils;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import net.micode.notes.data.Notes;
|
||||||
|
import net.micode.notes.data.Notes.CallNote;
|
||||||
|
import net.micode.notes.data.Notes.DataColumns;
|
||||||
|
import net.micode.notes.data.Notes.DataConstants;
|
||||||
|
import net.micode.notes.data.Notes.NoteColumns;
|
||||||
|
import net.micode.notes.data.Notes.TextNote;
|
||||||
|
import net.micode.notes.tool.ResourceParser.NoteBgResources;
|
||||||
|
|
||||||
|
|
||||||
|
public class WorkingNote {
|
||||||
|
// Note for the working note
|
||||||
|
private Note mNote;
|
||||||
|
// Note Id
|
||||||
|
private long mNoteId;
|
||||||
|
// Note content
|
||||||
|
private String mContent;
|
||||||
|
// Note mode
|
||||||
|
private int mMode;
|
||||||
|
|
||||||
|
private long mAlertDate;
|
||||||
|
|
||||||
|
private long mModifiedDate;
|
||||||
|
|
||||||
|
private int mBgColorId;
|
||||||
|
|
||||||
|
private int mWidgetId;
|
||||||
|
|
||||||
|
private int mWidgetType;
|
||||||
|
|
||||||
|
private long mFolderId;
|
||||||
|
|
||||||
|
private Context mContext;
|
||||||
|
|
||||||
|
private static final String TAG = "WorkingNote";
|
||||||
|
|
||||||
|
private boolean mIsDeleted;
|
||||||
|
|
||||||
|
private NoteSettingChangedListener mNoteSettingStatusListener;
|
||||||
|
|
||||||
|
public static final String[] DATA_PROJECTION = new String[] {
|
||||||
|
DataColumns.ID,
|
||||||
|
DataColumns.CONTENT,
|
||||||
|
DataColumns.MIME_TYPE,
|
||||||
|
DataColumns.DATA1,
|
||||||
|
DataColumns.DATA2,
|
||||||
|
DataColumns.DATA3,
|
||||||
|
DataColumns.DATA4,
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final String[] NOTE_PROJECTION = new String[] {
|
||||||
|
NoteColumns.PARENT_ID,
|
||||||
|
NoteColumns.ALERTED_DATE,
|
||||||
|
NoteColumns.BG_COLOR_ID,
|
||||||
|
NoteColumns.WIDGET_ID,
|
||||||
|
NoteColumns.WIDGET_TYPE,
|
||||||
|
NoteColumns.MODIFIED_DATE
|
||||||
|
};
|
||||||
|
|
||||||
|
private static final int DATA_ID_COLUMN = 0;
|
||||||
|
|
||||||
|
private static final int DATA_CONTENT_COLUMN = 1;
|
||||||
|
|
||||||
|
private static final int DATA_MIME_TYPE_COLUMN = 2;
|
||||||
|
|
||||||
|
private static final int DATA_MODE_COLUMN = 3;
|
||||||
|
|
||||||
|
private static final int NOTE_PARENT_ID_COLUMN = 0;
|
||||||
|
|
||||||
|
private static final int NOTE_ALERTED_DATE_COLUMN = 1;
|
||||||
|
|
||||||
|
private static final int NOTE_BG_COLOR_ID_COLUMN = 2;
|
||||||
|
|
||||||
|
private static final int NOTE_WIDGET_ID_COLUMN = 3;
|
||||||
|
|
||||||
|
private static final int NOTE_WIDGET_TYPE_COLUMN = 4;
|
||||||
|
|
||||||
|
private static final int NOTE_MODIFIED_DATE_COLUMN = 5;
|
||||||
|
|
||||||
|
// New note construct
|
||||||
|
private WorkingNote(Context context, long folderId) {
|
||||||
|
mContext = context;
|
||||||
|
mAlertDate = 0;
|
||||||
|
mModifiedDate = System.currentTimeMillis();
|
||||||
|
mFolderId = folderId;
|
||||||
|
mNote = new Note();
|
||||||
|
mNoteId = 0;
|
||||||
|
mIsDeleted = false;
|
||||||
|
mMode = 0;
|
||||||
|
mWidgetType = Notes.TYPE_WIDGET_INVALIDE;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Existing note construct
|
||||||
|
private WorkingNote(Context context, long noteId, long folderId) {
|
||||||
|
mContext = context;
|
||||||
|
mNoteId = noteId;
|
||||||
|
mFolderId = folderId;
|
||||||
|
mIsDeleted = false;
|
||||||
|
mNote = new Note();
|
||||||
|
loadNote();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loadNote() {
|
||||||
|
Cursor cursor = mContext.getContentResolver().query(
|
||||||
|
ContentUris.withAppendedId(Notes.CONTENT_NOTE_URI, mNoteId), NOTE_PROJECTION, null,
|
||||||
|
null, null);
|
||||||
|
|
||||||
|
if (cursor != null) {
|
||||||
|
if (cursor.moveToFirst()) {
|
||||||
|
mFolderId = cursor.getLong(NOTE_PARENT_ID_COLUMN);
|
||||||
|
mBgColorId = cursor.getInt(NOTE_BG_COLOR_ID_COLUMN);
|
||||||
|
mWidgetId = cursor.getInt(NOTE_WIDGET_ID_COLUMN);
|
||||||
|
mWidgetType = cursor.getInt(NOTE_WIDGET_TYPE_COLUMN);
|
||||||
|
mAlertDate = cursor.getLong(NOTE_ALERTED_DATE_COLUMN);
|
||||||
|
mModifiedDate = cursor.getLong(NOTE_MODIFIED_DATE_COLUMN);
|
||||||
|
}
|
||||||
|
cursor.close();
|
||||||
|
} else {
|
||||||
|
Log.e(TAG, "No note with id:" + mNoteId);
|
||||||
|
throw new IllegalArgumentException("Unable to find note with id " + mNoteId);
|
||||||
|
}
|
||||||
|
loadNoteData();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loadNoteData() {
|
||||||
|
Cursor cursor = mContext.getContentResolver().query(Notes.CONTENT_DATA_URI, DATA_PROJECTION,
|
||||||
|
DataColumns.NOTE_ID + "=?", new String[] {
|
||||||
|
String.valueOf(mNoteId)
|
||||||
|
}, null);
|
||||||
|
|
||||||
|
if (cursor != null) {
|
||||||
|
if (cursor.moveToFirst()) {
|
||||||
|
do {
|
||||||
|
String type = cursor.getString(DATA_MIME_TYPE_COLUMN);
|
||||||
|
if (DataConstants.NOTE.equals(type)) {
|
||||||
|
mContent = cursor.getString(DATA_CONTENT_COLUMN);
|
||||||
|
mMode = cursor.getInt(DATA_MODE_COLUMN);
|
||||||
|
mNote.setTextDataId(cursor.getLong(DATA_ID_COLUMN));
|
||||||
|
} else if (DataConstants.CALL_NOTE.equals(type)) {
|
||||||
|
mNote.setCallDataId(cursor.getLong(DATA_ID_COLUMN));
|
||||||
|
} else {
|
||||||
|
Log.d(TAG, "Wrong note type with type:" + type);
|
||||||
|
}
|
||||||
|
} while (cursor.moveToNext());
|
||||||
|
}
|
||||||
|
cursor.close();
|
||||||
|
} else {
|
||||||
|
Log.e(TAG, "No data with id:" + mNoteId);
|
||||||
|
throw new IllegalArgumentException("Unable to find note's data with id " + mNoteId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static WorkingNote createEmptyNote(Context context, long folderId, int widgetId,
|
||||||
|
int widgetType, int defaultBgColorId) {
|
||||||
|
WorkingNote note = new WorkingNote(context, folderId);
|
||||||
|
note.setBgColorId(defaultBgColorId);
|
||||||
|
note.setWidgetId(widgetId);
|
||||||
|
note.setWidgetType(widgetType);
|
||||||
|
return note;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static WorkingNote load(Context context, long id) {
|
||||||
|
return new WorkingNote(context, id, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public synchronized boolean saveNote() {
|
||||||
|
if (isWorthSaving()) {
|
||||||
|
if (!existInDatabase()) {
|
||||||
|
if ((mNoteId = Note.getNewNoteId(mContext, mFolderId)) == 0) {
|
||||||
|
Log.e(TAG, "Create new note fail with id:" + mNoteId);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mNote.syncNote(mContext, mNoteId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update widget content if there exist any widget of this note
|
||||||
|
*/
|
||||||
|
if (mWidgetId != AppWidgetManager.INVALID_APPWIDGET_ID
|
||||||
|
&& mWidgetType != Notes.TYPE_WIDGET_INVALIDE
|
||||||
|
&& mNoteSettingStatusListener != null) {
|
||||||
|
mNoteSettingStatusListener.onWidgetChanged();
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean existInDatabase() {
|
||||||
|
return mNoteId > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isWorthSaving() {
|
||||||
|
if (mIsDeleted || (!existInDatabase() && TextUtils.isEmpty(mContent))
|
||||||
|
|| (existInDatabase() && !mNote.isLocalModified())) {
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setOnSettingStatusChangedListener(NoteSettingChangedListener l) {
|
||||||
|
mNoteSettingStatusListener = l;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAlertDate(long date, boolean set) {
|
||||||
|
if (date != mAlertDate) {
|
||||||
|
mAlertDate = date;
|
||||||
|
mNote.setNoteValue(NoteColumns.ALERTED_DATE, String.valueOf(mAlertDate));
|
||||||
|
}
|
||||||
|
if (mNoteSettingStatusListener != null) {
|
||||||
|
mNoteSettingStatusListener.onClockAlertChanged(date, set);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void markDeleted(boolean mark) {
|
||||||
|
mIsDeleted = mark;
|
||||||
|
if (mWidgetId != AppWidgetManager.INVALID_APPWIDGET_ID
|
||||||
|
&& mWidgetType != Notes.TYPE_WIDGET_INVALIDE && mNoteSettingStatusListener != null) {
|
||||||
|
mNoteSettingStatusListener.onWidgetChanged();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setBgColorId(int id) {
|
||||||
|
if (id != mBgColorId) {
|
||||||
|
mBgColorId = id;
|
||||||
|
if (mNoteSettingStatusListener != null) {
|
||||||
|
mNoteSettingStatusListener.onBackgroundColorChanged();
|
||||||
|
}
|
||||||
|
mNote.setNoteValue(NoteColumns.BG_COLOR_ID, String.valueOf(id));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCheckListMode(int mode) {
|
||||||
|
if (mMode != mode) {
|
||||||
|
if (mNoteSettingStatusListener != null) {
|
||||||
|
mNoteSettingStatusListener.onCheckListModeChanged(mMode, mode);
|
||||||
|
}
|
||||||
|
mMode = mode;
|
||||||
|
mNote.setTextData(TextNote.MODE, String.valueOf(mMode));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setWidgetType(int type) {
|
||||||
|
if (type != mWidgetType) {
|
||||||
|
mWidgetType = type;
|
||||||
|
mNote.setNoteValue(NoteColumns.WIDGET_TYPE, String.valueOf(mWidgetType));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setWidgetId(int id) {
|
||||||
|
if (id != mWidgetId) {
|
||||||
|
mWidgetId = id;
|
||||||
|
mNote.setNoteValue(NoteColumns.WIDGET_ID, String.valueOf(mWidgetId));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setWorkingText(String text) {
|
||||||
|
if (!TextUtils.equals(mContent, text)) {
|
||||||
|
mContent = text;
|
||||||
|
mNote.setTextData(DataColumns.CONTENT, mContent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void convertToCallNote(String phoneNumber, long callDate) {
|
||||||
|
mNote.setCallData(CallNote.CALL_DATE, String.valueOf(callDate));
|
||||||
|
mNote.setCallData(CallNote.PHONE_NUMBER, phoneNumber);
|
||||||
|
mNote.setNoteValue(NoteColumns.PARENT_ID, String.valueOf(Notes.ID_CALL_RECORD_FOLDER));
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean hasClockAlert() {
|
||||||
|
return (mAlertDate > 0 ? true : false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getContent() {
|
||||||
|
return mContent;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getAlertDate() {
|
||||||
|
return mAlertDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getModifiedDate() {
|
||||||
|
return mModifiedDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getBgColorResId() {
|
||||||
|
return NoteBgResources.getNoteBgResource(mBgColorId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getBgColorId() {
|
||||||
|
return mBgColorId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getTitleBgResId() {
|
||||||
|
return NoteBgResources.getNoteTitleBgResource(mBgColorId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getCheckListMode() {
|
||||||
|
return mMode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getNoteId() {
|
||||||
|
return mNoteId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getFolderId() {
|
||||||
|
return mFolderId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getWidgetId() {
|
||||||
|
return mWidgetId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getWidgetType() {
|
||||||
|
return mWidgetType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface NoteSettingChangedListener {
|
||||||
|
/**
|
||||||
|
* Called when the background color of current note has just changed
|
||||||
|
*/
|
||||||
|
void onBackgroundColorChanged();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called when user set clock
|
||||||
|
*/
|
||||||
|
void onClockAlertChanged(long date, boolean set);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call when user create note from widget
|
||||||
|
*/
|
||||||
|
void onWidgetChanged();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call when switch between check list mode and normal mode
|
||||||
|
* @param oldMode is previous mode before change
|
||||||
|
* @param newMode is new mode
|
||||||
|
*/
|
||||||
|
void onCheckListModeChanged(int oldMode, int newMode);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,344 @@
|
|||||||
|
/*
|
||||||
|
* 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.tool;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.database.Cursor;
|
||||||
|
import android.os.Environment;
|
||||||
|
import android.text.TextUtils;
|
||||||
|
import android.text.format.DateFormat;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import net.micode.notes.R;
|
||||||
|
import net.micode.notes.data.Notes;
|
||||||
|
import net.micode.notes.data.Notes.DataColumns;
|
||||||
|
import net.micode.notes.data.Notes.DataConstants;
|
||||||
|
import net.micode.notes.data.Notes.NoteColumns;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.PrintStream;
|
||||||
|
|
||||||
|
|
||||||
|
public class BackupUtils {
|
||||||
|
private static final String TAG = "BackupUtils";
|
||||||
|
// Singleton stuff
|
||||||
|
private static BackupUtils sInstance;
|
||||||
|
|
||||||
|
public static synchronized BackupUtils getInstance(Context context) {
|
||||||
|
if (sInstance == null) {
|
||||||
|
sInstance = new BackupUtils(context);
|
||||||
|
}
|
||||||
|
return sInstance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Following states are signs to represents backup or restore
|
||||||
|
* status
|
||||||
|
*/
|
||||||
|
// Currently, the sdcard is not mounted
|
||||||
|
public static final int STATE_SD_CARD_UNMOUONTED = 0;
|
||||||
|
// The backup file not exist
|
||||||
|
public static final int STATE_BACKUP_FILE_NOT_EXIST = 1;
|
||||||
|
// The data is not well formated, may be changed by other programs
|
||||||
|
public static final int STATE_DATA_DESTROIED = 2;
|
||||||
|
// Some run-time exception which causes restore or backup fails
|
||||||
|
public static final int STATE_SYSTEM_ERROR = 3;
|
||||||
|
// Backup or restore success
|
||||||
|
public static final int STATE_SUCCESS = 4;
|
||||||
|
|
||||||
|
private TextExport mTextExport;
|
||||||
|
|
||||||
|
private BackupUtils(Context context) {
|
||||||
|
mTextExport = new TextExport(context);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean externalStorageAvailable() {
|
||||||
|
return Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState());
|
||||||
|
}
|
||||||
|
|
||||||
|
public int exportToText() {
|
||||||
|
return mTextExport.exportToText();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getExportedTextFileName() {
|
||||||
|
return mTextExport.mFileName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getExportedTextFileDir() {
|
||||||
|
return mTextExport.mFileDirectory;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TextExport {
|
||||||
|
private static final String[] NOTE_PROJECTION = {
|
||||||
|
NoteColumns.ID,
|
||||||
|
NoteColumns.MODIFIED_DATE,
|
||||||
|
NoteColumns.SNIPPET,
|
||||||
|
NoteColumns.TYPE
|
||||||
|
};
|
||||||
|
|
||||||
|
private static final int NOTE_COLUMN_ID = 0;
|
||||||
|
|
||||||
|
private static final int NOTE_COLUMN_MODIFIED_DATE = 1;
|
||||||
|
|
||||||
|
private static final int NOTE_COLUMN_SNIPPET = 2;
|
||||||
|
|
||||||
|
private static final String[] DATA_PROJECTION = {
|
||||||
|
DataColumns.CONTENT,
|
||||||
|
DataColumns.MIME_TYPE,
|
||||||
|
DataColumns.DATA1,
|
||||||
|
DataColumns.DATA2,
|
||||||
|
DataColumns.DATA3,
|
||||||
|
DataColumns.DATA4,
|
||||||
|
};
|
||||||
|
|
||||||
|
private static final int DATA_COLUMN_CONTENT = 0;
|
||||||
|
|
||||||
|
private static final int DATA_COLUMN_MIME_TYPE = 1;
|
||||||
|
|
||||||
|
private static final int DATA_COLUMN_CALL_DATE = 2;
|
||||||
|
|
||||||
|
private static final int DATA_COLUMN_PHONE_NUMBER = 4;
|
||||||
|
|
||||||
|
private final String [] TEXT_FORMAT;
|
||||||
|
private static final int FORMAT_FOLDER_NAME = 0;
|
||||||
|
private static final int FORMAT_NOTE_DATE = 1;
|
||||||
|
private static final int FORMAT_NOTE_CONTENT = 2;
|
||||||
|
|
||||||
|
private Context mContext;
|
||||||
|
private String mFileName;
|
||||||
|
private String mFileDirectory;
|
||||||
|
|
||||||
|
public TextExport(Context context) {
|
||||||
|
TEXT_FORMAT = context.getResources().getStringArray(R.array.format_for_exported_note);
|
||||||
|
mContext = context;
|
||||||
|
mFileName = "";
|
||||||
|
mFileDirectory = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getFormat(int id) {
|
||||||
|
return TEXT_FORMAT[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Export the folder identified by folder id to text
|
||||||
|
*/
|
||||||
|
private void exportFolderToText(String folderId, PrintStream ps) {
|
||||||
|
// Query notes belong to this folder
|
||||||
|
Cursor notesCursor = mContext.getContentResolver().query(Notes.CONTENT_NOTE_URI,
|
||||||
|
NOTE_PROJECTION, NoteColumns.PARENT_ID + "=?", new String[] {
|
||||||
|
folderId
|
||||||
|
}, null);
|
||||||
|
|
||||||
|
if (notesCursor != null) {
|
||||||
|
if (notesCursor.moveToFirst()) {
|
||||||
|
do {
|
||||||
|
// Print note's last modified date
|
||||||
|
ps.println(String.format(getFormat(FORMAT_NOTE_DATE), DateFormat.format(
|
||||||
|
mContext.getString(R.string.format_datetime_mdhm),
|
||||||
|
notesCursor.getLong(NOTE_COLUMN_MODIFIED_DATE))));
|
||||||
|
// Query data belong to this note
|
||||||
|
String noteId = notesCursor.getString(NOTE_COLUMN_ID);
|
||||||
|
exportNoteToText(noteId, ps);
|
||||||
|
} while (notesCursor.moveToNext());
|
||||||
|
}
|
||||||
|
notesCursor.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Export note identified by id to a print stream
|
||||||
|
*/
|
||||||
|
private void exportNoteToText(String noteId, PrintStream ps) {
|
||||||
|
Cursor dataCursor = mContext.getContentResolver().query(Notes.CONTENT_DATA_URI,
|
||||||
|
DATA_PROJECTION, DataColumns.NOTE_ID + "=?", new String[] {
|
||||||
|
noteId
|
||||||
|
}, null);
|
||||||
|
|
||||||
|
if (dataCursor != null) {
|
||||||
|
if (dataCursor.moveToFirst()) {
|
||||||
|
do {
|
||||||
|
String mimeType = dataCursor.getString(DATA_COLUMN_MIME_TYPE);
|
||||||
|
if (DataConstants.CALL_NOTE.equals(mimeType)) {
|
||||||
|
// Print phone number
|
||||||
|
String phoneNumber = dataCursor.getString(DATA_COLUMN_PHONE_NUMBER);
|
||||||
|
long callDate = dataCursor.getLong(DATA_COLUMN_CALL_DATE);
|
||||||
|
String location = dataCursor.getString(DATA_COLUMN_CONTENT);
|
||||||
|
|
||||||
|
if (!TextUtils.isEmpty(phoneNumber)) {
|
||||||
|
ps.println(String.format(getFormat(FORMAT_NOTE_CONTENT),
|
||||||
|
phoneNumber));
|
||||||
|
}
|
||||||
|
// Print call date
|
||||||
|
ps.println(String.format(getFormat(FORMAT_NOTE_CONTENT), DateFormat
|
||||||
|
.format(mContext.getString(R.string.format_datetime_mdhm),
|
||||||
|
callDate)));
|
||||||
|
// Print call attachment location
|
||||||
|
if (!TextUtils.isEmpty(location)) {
|
||||||
|
ps.println(String.format(getFormat(FORMAT_NOTE_CONTENT),
|
||||||
|
location));
|
||||||
|
}
|
||||||
|
} else if (DataConstants.NOTE.equals(mimeType)) {
|
||||||
|
String content = dataCursor.getString(DATA_COLUMN_CONTENT);
|
||||||
|
if (!TextUtils.isEmpty(content)) {
|
||||||
|
ps.println(String.format(getFormat(FORMAT_NOTE_CONTENT),
|
||||||
|
content));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} while (dataCursor.moveToNext());
|
||||||
|
}
|
||||||
|
dataCursor.close();
|
||||||
|
}
|
||||||
|
// print a line separator between note
|
||||||
|
try {
|
||||||
|
ps.write(new byte[] {
|
||||||
|
Character.LINE_SEPARATOR, Character.LETTER_NUMBER
|
||||||
|
});
|
||||||
|
} catch (IOException e) {
|
||||||
|
Log.e(TAG, e.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Note will be exported as text which is user readable
|
||||||
|
*/
|
||||||
|
public int exportToText() {
|
||||||
|
if (!externalStorageAvailable()) {
|
||||||
|
Log.d(TAG, "Media was not mounted");
|
||||||
|
return STATE_SD_CARD_UNMOUONTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintStream ps = getExportToTextPrintStream();
|
||||||
|
if (ps == null) {
|
||||||
|
Log.e(TAG, "get print stream error");
|
||||||
|
return STATE_SYSTEM_ERROR;
|
||||||
|
}
|
||||||
|
// First export folder and its notes
|
||||||
|
Cursor folderCursor = mContext.getContentResolver().query(
|
||||||
|
Notes.CONTENT_NOTE_URI,
|
||||||
|
NOTE_PROJECTION,
|
||||||
|
"(" + NoteColumns.TYPE + "=" + Notes.TYPE_FOLDER + " AND "
|
||||||
|
+ NoteColumns.PARENT_ID + "<>" + Notes.ID_TRASH_FOLER + ") OR "
|
||||||
|
+ NoteColumns.ID + "=" + Notes.ID_CALL_RECORD_FOLDER, null, null);
|
||||||
|
|
||||||
|
if (folderCursor != null) {
|
||||||
|
if (folderCursor.moveToFirst()) {
|
||||||
|
do {
|
||||||
|
// Print folder's name
|
||||||
|
String folderName = "";
|
||||||
|
if(folderCursor.getLong(NOTE_COLUMN_ID) == Notes.ID_CALL_RECORD_FOLDER) {
|
||||||
|
folderName = mContext.getString(R.string.call_record_folder_name);
|
||||||
|
} else {
|
||||||
|
folderName = folderCursor.getString(NOTE_COLUMN_SNIPPET);
|
||||||
|
}
|
||||||
|
if (!TextUtils.isEmpty(folderName)) {
|
||||||
|
ps.println(String.format(getFormat(FORMAT_FOLDER_NAME), folderName));
|
||||||
|
}
|
||||||
|
String folderId = folderCursor.getString(NOTE_COLUMN_ID);
|
||||||
|
exportFolderToText(folderId, ps);
|
||||||
|
} while (folderCursor.moveToNext());
|
||||||
|
}
|
||||||
|
folderCursor.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Export notes in root's folder
|
||||||
|
Cursor noteCursor = mContext.getContentResolver().query(
|
||||||
|
Notes.CONTENT_NOTE_URI,
|
||||||
|
NOTE_PROJECTION,
|
||||||
|
NoteColumns.TYPE + "=" + +Notes.TYPE_NOTE + " AND " + NoteColumns.PARENT_ID
|
||||||
|
+ "=0", null, null);
|
||||||
|
|
||||||
|
if (noteCursor != null) {
|
||||||
|
if (noteCursor.moveToFirst()) {
|
||||||
|
do {
|
||||||
|
ps.println(String.format(getFormat(FORMAT_NOTE_DATE), DateFormat.format(
|
||||||
|
mContext.getString(R.string.format_datetime_mdhm),
|
||||||
|
noteCursor.getLong(NOTE_COLUMN_MODIFIED_DATE))));
|
||||||
|
// Query data belong to this note
|
||||||
|
String noteId = noteCursor.getString(NOTE_COLUMN_ID);
|
||||||
|
exportNoteToText(noteId, ps);
|
||||||
|
} while (noteCursor.moveToNext());
|
||||||
|
}
|
||||||
|
noteCursor.close();
|
||||||
|
}
|
||||||
|
ps.close();
|
||||||
|
|
||||||
|
return STATE_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a print stream pointed to the file {@generateExportedTextFile}
|
||||||
|
*/
|
||||||
|
private PrintStream getExportToTextPrintStream() {
|
||||||
|
File file = generateFileMountedOnSDcard(mContext, R.string.file_path,
|
||||||
|
R.string.file_name_txt_format);
|
||||||
|
if (file == null) {
|
||||||
|
Log.e(TAG, "create file to exported failed");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
mFileName = file.getName();
|
||||||
|
mFileDirectory = mContext.getString(R.string.file_path);
|
||||||
|
PrintStream ps = null;
|
||||||
|
try {
|
||||||
|
FileOutputStream fos = new FileOutputStream(file);
|
||||||
|
ps = new PrintStream(fos);
|
||||||
|
} catch (FileNotFoundException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
return null;
|
||||||
|
} catch (NullPointerException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return ps;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate the text file to store imported data
|
||||||
|
*/
|
||||||
|
private static File generateFileMountedOnSDcard(Context context, int filePathResId, int fileNameFormatResId) {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append(Environment.getExternalStorageDirectory());
|
||||||
|
sb.append(context.getString(filePathResId));
|
||||||
|
File filedir = new File(sb.toString());
|
||||||
|
sb.append(context.getString(
|
||||||
|
fileNameFormatResId,
|
||||||
|
DateFormat.format(context.getString(R.string.format_date_ymd),
|
||||||
|
System.currentTimeMillis())));
|
||||||
|
File file = new File(sb.toString());
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (!filedir.exists()) {
|
||||||
|
filedir.mkdir();
|
||||||
|
}
|
||||||
|
if (!file.exists()) {
|
||||||
|
file.createNewFile();
|
||||||
|
}
|
||||||
|
return file;
|
||||||
|
} catch (SecurityException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
} catch (IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -0,0 +1,181 @@
|
|||||||
|
/*
|
||||||
|
* 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.tool;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.preference.PreferenceManager;
|
||||||
|
|
||||||
|
import net.micode.notes.R;
|
||||||
|
import net.micode.notes.ui.NotesPreferenceActivity;
|
||||||
|
|
||||||
|
public class ResourceParser {
|
||||||
|
|
||||||
|
public static final int YELLOW = 0;
|
||||||
|
public static final int BLUE = 1;
|
||||||
|
public static final int WHITE = 2;
|
||||||
|
public static final int GREEN = 3;
|
||||||
|
public static final int RED = 4;
|
||||||
|
|
||||||
|
public static final int BG_DEFAULT_COLOR = YELLOW;
|
||||||
|
|
||||||
|
public static final int TEXT_SMALL = 0;
|
||||||
|
public static final int TEXT_MEDIUM = 1;
|
||||||
|
public static final int TEXT_LARGE = 2;
|
||||||
|
public static final int TEXT_SUPER = 3;
|
||||||
|
|
||||||
|
public static final int BG_DEFAULT_FONT_SIZE = TEXT_MEDIUM;
|
||||||
|
|
||||||
|
public static class NoteBgResources {
|
||||||
|
private final static int [] BG_EDIT_RESOURCES = new int [] {
|
||||||
|
R.drawable.edit_yellow,
|
||||||
|
R.drawable.edit_blue,
|
||||||
|
R.drawable.edit_white,
|
||||||
|
R.drawable.edit_green,
|
||||||
|
R.drawable.edit_red
|
||||||
|
};
|
||||||
|
|
||||||
|
private final static int [] BG_EDIT_TITLE_RESOURCES = new int [] {
|
||||||
|
R.drawable.edit_title_yellow,
|
||||||
|
R.drawable.edit_title_blue,
|
||||||
|
R.drawable.edit_title_white,
|
||||||
|
R.drawable.edit_title_green,
|
||||||
|
R.drawable.edit_title_red
|
||||||
|
};
|
||||||
|
|
||||||
|
public static int getNoteBgResource(int id) {
|
||||||
|
return BG_EDIT_RESOURCES[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getNoteTitleBgResource(int id) {
|
||||||
|
return BG_EDIT_TITLE_RESOURCES[id];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getDefaultBgId(Context context) {
|
||||||
|
if (PreferenceManager.getDefaultSharedPreferences(context).getBoolean(
|
||||||
|
NotesPreferenceActivity.PREFERENCE_SET_BG_COLOR_KEY, false)) {
|
||||||
|
return (int) (Math.random() * NoteBgResources.BG_EDIT_RESOURCES.length);
|
||||||
|
} else {
|
||||||
|
return BG_DEFAULT_COLOR;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class NoteItemBgResources {
|
||||||
|
private final static int [] BG_FIRST_RESOURCES = new int [] {
|
||||||
|
R.drawable.list_yellow_up,
|
||||||
|
R.drawable.list_blue_up,
|
||||||
|
R.drawable.list_white_up,
|
||||||
|
R.drawable.list_green_up,
|
||||||
|
R.drawable.list_red_up
|
||||||
|
};
|
||||||
|
|
||||||
|
private final static int [] BG_NORMAL_RESOURCES = new int [] {
|
||||||
|
R.drawable.list_yellow_middle,
|
||||||
|
R.drawable.list_blue_middle,
|
||||||
|
R.drawable.list_white_middle,
|
||||||
|
R.drawable.list_green_middle,
|
||||||
|
R.drawable.list_red_middle
|
||||||
|
};
|
||||||
|
|
||||||
|
private final static int [] BG_LAST_RESOURCES = new int [] {
|
||||||
|
R.drawable.list_yellow_down,
|
||||||
|
R.drawable.list_blue_down,
|
||||||
|
R.drawable.list_white_down,
|
||||||
|
R.drawable.list_green_down,
|
||||||
|
R.drawable.list_red_down,
|
||||||
|
};
|
||||||
|
|
||||||
|
private final static int [] BG_SINGLE_RESOURCES = new int [] {
|
||||||
|
R.drawable.list_yellow_single,
|
||||||
|
R.drawable.list_blue_single,
|
||||||
|
R.drawable.list_white_single,
|
||||||
|
R.drawable.list_green_single,
|
||||||
|
R.drawable.list_red_single
|
||||||
|
};
|
||||||
|
|
||||||
|
public static int getNoteBgFirstRes(int id) {
|
||||||
|
return BG_FIRST_RESOURCES[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getNoteBgLastRes(int id) {
|
||||||
|
return BG_LAST_RESOURCES[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getNoteBgSingleRes(int id) {
|
||||||
|
return BG_SINGLE_RESOURCES[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getNoteBgNormalRes(int id) {
|
||||||
|
return BG_NORMAL_RESOURCES[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getFolderBgRes() {
|
||||||
|
return R.drawable.list_folder;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class WidgetBgResources {
|
||||||
|
private final static int [] BG_2X_RESOURCES = new int [] {
|
||||||
|
R.drawable.widget_2x_yellow,
|
||||||
|
R.drawable.widget_2x_blue,
|
||||||
|
R.drawable.widget_2x_white,
|
||||||
|
R.drawable.widget_2x_green,
|
||||||
|
R.drawable.widget_2x_red,
|
||||||
|
};
|
||||||
|
|
||||||
|
public static int getWidget2xBgResource(int id) {
|
||||||
|
return BG_2X_RESOURCES[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
private final static int [] BG_4X_RESOURCES = new int [] {
|
||||||
|
R.drawable.widget_4x_yellow,
|
||||||
|
R.drawable.widget_4x_blue,
|
||||||
|
R.drawable.widget_4x_white,
|
||||||
|
R.drawable.widget_4x_green,
|
||||||
|
R.drawable.widget_4x_red
|
||||||
|
};
|
||||||
|
|
||||||
|
public static int getWidget4xBgResource(int id) {
|
||||||
|
return BG_4X_RESOURCES[id];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class TextAppearanceResources {
|
||||||
|
private final static int [] TEXTAPPEARANCE_RESOURCES = new int [] {
|
||||||
|
R.style.TextAppearanceNormal,
|
||||||
|
R.style.TextAppearanceMedium,
|
||||||
|
R.style.TextAppearanceLarge,
|
||||||
|
R.style.TextAppearanceSuper
|
||||||
|
};
|
||||||
|
|
||||||
|
public static int getTexAppearanceResource(int id) {
|
||||||
|
/**
|
||||||
|
* HACKME: Fix bug of store the resource id in shared preference.
|
||||||
|
* The id may larger than the length of resources, in this case,
|
||||||
|
* return the {@link ResourceParser#BG_DEFAULT_FONT_SIZE}
|
||||||
|
*/
|
||||||
|
if (id >= TEXTAPPEARANCE_RESOURCES.length) {
|
||||||
|
return BG_DEFAULT_FONT_SIZE;
|
||||||
|
}
|
||||||
|
return TEXTAPPEARANCE_RESOURCES[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getResourcesSize() {
|
||||||
|
return TEXTAPPEARANCE_RESOURCES.length;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -1,67 +0,0 @@
|
|||||||
/*
|
|
||||||
* 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.content.Context;
|
|
||||||
import android.view.Menu;
|
|
||||||
import android.view.MenuItem;
|
|
||||||
import android.view.View;
|
|
||||||
import android.view.View.OnClickListener;
|
|
||||||
import android.widget.Button;
|
|
||||||
import android.widget.PopupMenu;
|
|
||||||
import android.widget.PopupMenu.OnMenuItemClickListener;
|
|
||||||
|
|
||||||
import net.micode.notes.R;
|
|
||||||
|
|
||||||
public class DropdownMenu {
|
|
||||||
private Button mButton;
|
|
||||||
private PopupMenu mPopupMenu;
|
|
||||||
//声明一个下拉菜单
|
|
||||||
private Menu mMenu;
|
|
||||||
|
|
||||||
public DropdownMenu(Context context, Button button, int menuId) {
|
|
||||||
mButton = button;
|
|
||||||
mButton.setBackgroundResource(R.drawable.dropdown_icon);
|
|
||||||
//设置这个view的背景
|
|
||||||
mPopupMenu = new PopupMenu(context, mButton);
|
|
||||||
mMenu = mPopupMenu.getMenu();
|
|
||||||
mPopupMenu.getMenuInflater().inflate(menuId, mMenu);
|
|
||||||
//MenuInflater是用来实例化Menu目录下的Menu布局文件
|
|
||||||
//根据ID来确认menu的内容选项
|
|
||||||
mButton.setOnClickListener(new OnClickListener() {
|
|
||||||
public void onClick(View v) {
|
|
||||||
mPopupMenu.show();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setOnDropdownMenuItemClickListener(OnMenuItemClickListener listener) {
|
|
||||||
if (mPopupMenu != null) {
|
|
||||||
mPopupMenu.setOnMenuItemClickListener(listener);
|
|
||||||
}//设置菜单的监听
|
|
||||||
}
|
|
||||||
|
|
||||||
public MenuItem findItem(int id) {
|
|
||||||
return mMenu.findItem(id);
|
|
||||||
}
|
|
||||||
//对于菜单选项的初始化,根据索引搜索菜单需要的选项
|
|
||||||
|
|
||||||
public void setTitle(CharSequence title) {
|
|
||||||
mButton.setText(title);
|
|
||||||
}
|
|
||||||
//布局文件,设置标题
|
|
||||||
}
|
|
||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -1,515 +0,0 @@
|
|||||||
/*
|
|
||||||
* 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.appwidget.AppWidgetManager;
|
|
||||||
import android.content.ContentResolver;
|
|
||||||
import android.content.ContentValues;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.database.Cursor;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import net.micode.notes.data.Notes;
|
|
||||||
import net.micode.notes.data.Notes.DataColumns;
|
|
||||||
import net.micode.notes.data.Notes.NoteColumns;
|
|
||||||
import net.micode.notes.gtask.exception.ActionFailureException;
|
|
||||||
import net.micode.notes.tool.GTaskStringUtils;
|
|
||||||
import net.micode.notes.tool.ResourceParser;
|
|
||||||
|
|
||||||
import org.json.JSONArray;
|
|
||||||
import org.json.JSONException;
|
|
||||||
import org.json.JSONObject;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
|
|
||||||
// 定义 SqlNote 类
|
|
||||||
public class SqlNote {
|
|
||||||
// 定义日志标签,用于调试和输出日志信息
|
|
||||||
private static final String TAG = SqlNote.class.getSimpleName();
|
|
||||||
// 定义一个无效的 ID 常量
|
|
||||||
private static final int INVALID_ID = -99999;
|
|
||||||
// 定义 PROJECTION_NOTE 数组,用于查询数据库时指定需要的列
|
|
||||||
public static final String[] PROJECTION_NOTE = new String[] {
|
|
||||||
NoteColumns.ID, NoteColumns.ALERTED_DATE, NoteColumns.BG_COLOR_ID,
|
|
||||||
NoteColumns.CREATED_DATE, NoteColumns.HAS_ATTACHMENT, NoteColumns.MODIFIED_DATE,
|
|
||||||
NoteColumns.NOTES_COUNT, NoteColumns.PARENT_ID, NoteColumns.SNIPPET, NoteColumns.TYPE,
|
|
||||||
NoteColumns.WIDGET_ID, NoteColumns.WIDGET_TYPE, NoteColumns.SYNC_ID,
|
|
||||||
NoteColumns.LOCAL_MODIFIED, NoteColumns.ORIGIN_PARENT_ID, NoteColumns.GTASK_ID,
|
|
||||||
NoteColumns.VERSION
|
|
||||||
};
|
|
||||||
|
|
||||||
public static final int ID_COLUMN = 0;
|
|
||||||
|
|
||||||
public static final int ALERTED_DATE_COLUMN = 1;
|
|
||||||
|
|
||||||
public static final int BG_COLOR_ID_COLUMN = 2;
|
|
||||||
|
|
||||||
public static final int CREATED_DATE_COLUMN = 3;
|
|
||||||
|
|
||||||
public static final int HAS_ATTACHMENT_COLUMN = 4;
|
|
||||||
|
|
||||||
public static final int MODIFIED_DATE_COLUMN = 5;
|
|
||||||
|
|
||||||
public static final int NOTES_COUNT_COLUMN = 6;
|
|
||||||
|
|
||||||
public static final int PARENT_ID_COLUMN = 7;
|
|
||||||
|
|
||||||
public static final int SNIPPET_COLUMN = 8;
|
|
||||||
|
|
||||||
public static final int TYPE_COLUMN = 9;
|
|
||||||
|
|
||||||
public static final int WIDGET_ID_COLUMN = 10;
|
|
||||||
|
|
||||||
public static final int WIDGET_TYPE_COLUMN = 11;
|
|
||||||
|
|
||||||
public static final int SYNC_ID_COLUMN = 12;
|
|
||||||
|
|
||||||
public static final int LOCAL_MODIFIED_COLUMN = 13;
|
|
||||||
|
|
||||||
public static final int ORIGIN_PARENT_ID_COLUMN = 14;
|
|
||||||
|
|
||||||
public static final int GTASK_ID_COLUMN = 15;
|
|
||||||
|
|
||||||
public static final int VERSION_COLUMN = 16;
|
|
||||||
|
|
||||||
private Context mContext;
|
|
||||||
|
|
||||||
private ContentResolver mContentResolver;
|
|
||||||
|
|
||||||
private boolean mIsCreate;
|
|
||||||
|
|
||||||
private long mId;
|
|
||||||
|
|
||||||
private long mAlertDate;
|
|
||||||
|
|
||||||
private int mBgColorId;
|
|
||||||
|
|
||||||
private long mCreatedDate;
|
|
||||||
|
|
||||||
private int mHasAttachment;
|
|
||||||
|
|
||||||
private long mModifiedDate;
|
|
||||||
|
|
||||||
private long mParentId;
|
|
||||||
|
|
||||||
private String mSnippet;
|
|
||||||
|
|
||||||
private int mType;
|
|
||||||
|
|
||||||
private int mWidgetId;
|
|
||||||
|
|
||||||
private int mWidgetType;
|
|
||||||
|
|
||||||
private long mOriginParent;
|
|
||||||
|
|
||||||
private long mVersion;
|
|
||||||
|
|
||||||
private ContentValues mDiffNoteValues;
|
|
||||||
|
|
||||||
private ArrayList<SqlData> mDataList;
|
|
||||||
|
|
||||||
// 构造方法:用于创建新的笔记对象
|
|
||||||
public SqlNote(Context context) {
|
|
||||||
mContext = context;
|
|
||||||
mContentResolver = context.getContentResolver();
|
|
||||||
mIsCreate = true;
|
|
||||||
mId = INVALID_ID;
|
|
||||||
mAlertDate = 0;
|
|
||||||
mBgColorId = ResourceParser.getDefaultBgId(context);
|
|
||||||
mCreatedDate = System.currentTimeMillis();
|
|
||||||
mHasAttachment = 0;
|
|
||||||
mModifiedDate = System.currentTimeMillis();
|
|
||||||
mParentId = 0;
|
|
||||||
mSnippet = "";
|
|
||||||
mType = Notes.TYPE_NOTE;
|
|
||||||
mWidgetId = AppWidgetManager.INVALID_APPWIDGET_ID;
|
|
||||||
mWidgetType = Notes.TYPE_WIDGET_INVALIDE;
|
|
||||||
mOriginParent = 0;
|
|
||||||
mVersion = 0;
|
|
||||||
mDiffNoteValues = new ContentValues();
|
|
||||||
mDataList = new ArrayList<SqlData>();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 构造方法:用于从数据库的 Cursor 加载笔记对象
|
|
||||||
public SqlNote(Context context, Cursor c) {
|
|
||||||
mContext = context;
|
|
||||||
mContentResolver = context.getContentResolver();
|
|
||||||
mIsCreate = false;
|
|
||||||
loadFromCursor(c);
|
|
||||||
mDataList = new ArrayList<SqlData>();
|
|
||||||
if (mType == Notes.TYPE_NOTE)
|
|
||||||
loadDataContent();
|
|
||||||
mDiffNoteValues = new ContentValues();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 构造方法:通过笔记 ID 从数据库加载笔记对象
|
|
||||||
public SqlNote(Context context, long id) {
|
|
||||||
mContext = context;
|
|
||||||
mContentResolver = context.getContentResolver();
|
|
||||||
mIsCreate = false;
|
|
||||||
loadFromCursor(id);
|
|
||||||
mDataList = new ArrayList<SqlData>();
|
|
||||||
if (mType == Notes.TYPE_NOTE)
|
|
||||||
loadDataContent();
|
|
||||||
mDiffNoteValues = new ContentValues();
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// 通过笔记 ID 从数据库加载笔记数据
|
|
||||||
private void loadFromCursor(long id) {
|
|
||||||
Cursor c = null;
|
|
||||||
try {
|
|
||||||
c = mContentResolver.query(Notes.CONTENT_NOTE_URI, PROJECTION_NOTE, "(_id=?)",
|
|
||||||
new String[] {
|
|
||||||
String.valueOf(id)
|
|
||||||
}, null);
|
|
||||||
if (c != null) {
|
|
||||||
c.moveToNext();
|
|
||||||
loadFromCursor(c);
|
|
||||||
} else {
|
|
||||||
Log.w(TAG, "loadFromCursor: cursor = null");
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (c != null)
|
|
||||||
c.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 从 Cursor 加载 SqlNote 对象的数据
|
|
||||||
private void loadFromCursor(Cursor c) {
|
|
||||||
mId = c.getLong(ID_COLUMN);
|
|
||||||
mAlertDate = c.getLong(ALERTED_DATE_COLUMN);
|
|
||||||
mBgColorId = c.getInt(BG_COLOR_ID_COLUMN);
|
|
||||||
mCreatedDate = c.getLong(CREATED_DATE_COLUMN);
|
|
||||||
mHasAttachment = c.getInt(HAS_ATTACHMENT_COLUMN);
|
|
||||||
mModifiedDate = c.getLong(MODIFIED_DATE_COLUMN);
|
|
||||||
mParentId = c.getLong(PARENT_ID_COLUMN);
|
|
||||||
mSnippet = c.getString(SNIPPET_COLUMN);
|
|
||||||
mType = c.getInt(TYPE_COLUMN);
|
|
||||||
mWidgetId = c.getInt(WIDGET_ID_COLUMN);
|
|
||||||
mWidgetType = c.getInt(WIDGET_TYPE_COLUMN);
|
|
||||||
mVersion = c.getLong(VERSION_COLUMN);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 加载笔记的数据内容
|
|
||||||
private void loadDataContent() {
|
|
||||||
Cursor c = null;
|
|
||||||
mDataList.clear();
|
|
||||||
try {
|
|
||||||
c = mContentResolver.query(Notes.CONTENT_DATA_URI, SqlData.PROJECTION_DATA,
|
|
||||||
"(note_id=?)", new String[] {
|
|
||||||
String.valueOf(mId)
|
|
||||||
}, null);
|
|
||||||
if (c != null) {
|
|
||||||
if (c.getCount() == 0) {
|
|
||||||
Log.w(TAG, "it seems that the note has not data");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
while (c.moveToNext()) {
|
|
||||||
SqlData data = new SqlData(mContext, c);
|
|
||||||
mDataList.add(data);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.w(TAG, "loadDataContent: cursor = null");
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (c != null)
|
|
||||||
c.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean setContent(JSONObject js) {
|
|
||||||
try {
|
|
||||||
// 获取包含笔记信息的 JSON 对象
|
|
||||||
JSONObject note = js.getJSONObject(GTaskStringUtils.META_HEAD_NOTE);
|
|
||||||
if (note.getInt(NoteColumns.TYPE) == Notes.TYPE_SYSTEM) {
|
|
||||||
Log.w(TAG, "cannot set system folder");
|
|
||||||
} else if (note.getInt(NoteColumns.TYPE) == Notes.TYPE_FOLDER) {
|
|
||||||
// for folder we can only update the snnipet and type
|
|
||||||
String snippet = note.has(NoteColumns.SNIPPET) ? note
|
|
||||||
.getString(NoteColumns.SNIPPET) : "";
|
|
||||||
if (mIsCreate || !mSnippet.equals(snippet)) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.SNIPPET, snippet);
|
|
||||||
}
|
|
||||||
mSnippet = snippet;
|
|
||||||
|
|
||||||
int type = note.has(NoteColumns.TYPE) ? note.getInt(NoteColumns.TYPE)
|
|
||||||
: Notes.TYPE_NOTE;
|
|
||||||
if (mIsCreate || mType != type) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.TYPE, type);
|
|
||||||
}
|
|
||||||
mType = type;
|
|
||||||
} else if (note.getInt(NoteColumns.TYPE) == Notes.TYPE_NOTE) {
|
|
||||||
JSONArray dataArray = js.getJSONArray(GTaskStringUtils.META_HEAD_DATA);
|
|
||||||
long id = note.has(NoteColumns.ID) ? note.getLong(NoteColumns.ID) : INVALID_ID;
|
|
||||||
if (mIsCreate || mId != id) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.ID, id);
|
|
||||||
}
|
|
||||||
mId = id;
|
|
||||||
|
|
||||||
long alertDate = note.has(NoteColumns.ALERTED_DATE) ? note
|
|
||||||
.getLong(NoteColumns.ALERTED_DATE) : 0;
|
|
||||||
if (mIsCreate || mAlertDate != alertDate) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.ALERTED_DATE, alertDate);
|
|
||||||
}
|
|
||||||
mAlertDate = alertDate;
|
|
||||||
|
|
||||||
int bgColorId = note.has(NoteColumns.BG_COLOR_ID) ? note
|
|
||||||
.getInt(NoteColumns.BG_COLOR_ID) : ResourceParser.getDefaultBgId(mContext);
|
|
||||||
if (mIsCreate || mBgColorId != bgColorId) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.BG_COLOR_ID, bgColorId);
|
|
||||||
}
|
|
||||||
mBgColorId = bgColorId;
|
|
||||||
|
|
||||||
long createDate = note.has(NoteColumns.CREATED_DATE) ? note
|
|
||||||
.getLong(NoteColumns.CREATED_DATE) : System.currentTimeMillis();
|
|
||||||
if (mIsCreate || mCreatedDate != createDate) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.CREATED_DATE, createDate);
|
|
||||||
}
|
|
||||||
mCreatedDate = createDate;
|
|
||||||
|
|
||||||
int hasAttachment = note.has(NoteColumns.HAS_ATTACHMENT) ? note
|
|
||||||
.getInt(NoteColumns.HAS_ATTACHMENT) : 0;
|
|
||||||
if (mIsCreate || mHasAttachment != hasAttachment) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.HAS_ATTACHMENT, hasAttachment);
|
|
||||||
}
|
|
||||||
mHasAttachment = hasAttachment;
|
|
||||||
|
|
||||||
long modifiedDate = note.has(NoteColumns.MODIFIED_DATE) ? note
|
|
||||||
.getLong(NoteColumns.MODIFIED_DATE) : System.currentTimeMillis();
|
|
||||||
if (mIsCreate || mModifiedDate != modifiedDate) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.MODIFIED_DATE, modifiedDate);
|
|
||||||
}
|
|
||||||
mModifiedDate = modifiedDate;
|
|
||||||
|
|
||||||
long parentId = note.has(NoteColumns.PARENT_ID) ? note
|
|
||||||
.getLong(NoteColumns.PARENT_ID) : 0;
|
|
||||||
if (mIsCreate || mParentId != parentId) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.PARENT_ID, parentId);
|
|
||||||
}
|
|
||||||
mParentId = parentId;
|
|
||||||
|
|
||||||
String snippet = note.has(NoteColumns.SNIPPET) ? note
|
|
||||||
.getString(NoteColumns.SNIPPET) : "";
|
|
||||||
if (mIsCreate || !mSnippet.equals(snippet)) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.SNIPPET, snippet);
|
|
||||||
}
|
|
||||||
mSnippet = snippet;
|
|
||||||
|
|
||||||
int type = note.has(NoteColumns.TYPE) ? note.getInt(NoteColumns.TYPE)
|
|
||||||
: Notes.TYPE_NOTE;
|
|
||||||
if (mIsCreate || mType != type) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.TYPE, type);
|
|
||||||
}
|
|
||||||
mType = type;
|
|
||||||
|
|
||||||
int widgetId = note.has(NoteColumns.WIDGET_ID) ? note.getInt(NoteColumns.WIDGET_ID)
|
|
||||||
: AppWidgetManager.INVALID_APPWIDGET_ID;
|
|
||||||
if (mIsCreate || mWidgetId != widgetId) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.WIDGET_ID, widgetId);
|
|
||||||
}
|
|
||||||
mWidgetId = widgetId;
|
|
||||||
|
|
||||||
int widgetType = note.has(NoteColumns.WIDGET_TYPE) ? note
|
|
||||||
.getInt(NoteColumns.WIDGET_TYPE) : Notes.TYPE_WIDGET_INVALIDE;
|
|
||||||
if (mIsCreate || mWidgetType != widgetType) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.WIDGET_TYPE, widgetType);
|
|
||||||
}
|
|
||||||
mWidgetType = widgetType;
|
|
||||||
|
|
||||||
long originParent = note.has(NoteColumns.ORIGIN_PARENT_ID) ? note
|
|
||||||
.getLong(NoteColumns.ORIGIN_PARENT_ID) : 0;
|
|
||||||
if (mIsCreate || mOriginParent != originParent) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.ORIGIN_PARENT_ID, originParent);
|
|
||||||
}
|
|
||||||
mOriginParent = originParent;
|
|
||||||
|
|
||||||
for (int i = 0; i < dataArray.length(); i++) {
|
|
||||||
JSONObject data = dataArray.getJSONObject(i);
|
|
||||||
SqlData sqlData = null;
|
|
||||||
if (data.has(DataColumns.ID)) {
|
|
||||||
long dataId = data.getLong(DataColumns.ID);
|
|
||||||
for (SqlData temp : mDataList) {
|
|
||||||
if (dataId == temp.getId()) {
|
|
||||||
sqlData = temp;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sqlData == null) {
|
|
||||||
sqlData = new SqlData(mContext);
|
|
||||||
mDataList.add(sqlData);
|
|
||||||
}
|
|
||||||
|
|
||||||
sqlData.setContent(data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (JSONException e) {
|
|
||||||
Log.e(TAG, e.toString());
|
|
||||||
e.printStackTrace();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public JSONObject getContent() {
|
|
||||||
try {
|
|
||||||
JSONObject js = new JSONObject();
|
|
||||||
|
|
||||||
if (mIsCreate) {
|
|
||||||
Log.e(TAG, "it seems that we haven't created this in database yet");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
JSONObject note = new JSONObject();
|
|
||||||
if (mType == Notes.TYPE_NOTE) {
|
|
||||||
note.put(NoteColumns.ID, mId);
|
|
||||||
note.put(NoteColumns.ALERTED_DATE, mAlertDate);
|
|
||||||
note.put(NoteColumns.BG_COLOR_ID, mBgColorId);
|
|
||||||
note.put(NoteColumns.CREATED_DATE, mCreatedDate);
|
|
||||||
note.put(NoteColumns.HAS_ATTACHMENT, mHasAttachment);
|
|
||||||
note.put(NoteColumns.MODIFIED_DATE, mModifiedDate);
|
|
||||||
note.put(NoteColumns.PARENT_ID, mParentId);
|
|
||||||
note.put(NoteColumns.SNIPPET, mSnippet);
|
|
||||||
note.put(NoteColumns.TYPE, mType);
|
|
||||||
note.put(NoteColumns.WIDGET_ID, mWidgetId);
|
|
||||||
note.put(NoteColumns.WIDGET_TYPE, mWidgetType);
|
|
||||||
note.put(NoteColumns.ORIGIN_PARENT_ID, mOriginParent);
|
|
||||||
js.put(GTaskStringUtils.META_HEAD_NOTE, note);
|
|
||||||
|
|
||||||
JSONArray dataArray = new JSONArray();
|
|
||||||
for (SqlData sqlData : mDataList) {
|
|
||||||
JSONObject data = sqlData.getContent();
|
|
||||||
if (data != null) {
|
|
||||||
dataArray.put(data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
js.put(GTaskStringUtils.META_HEAD_DATA, dataArray);
|
|
||||||
} else if (mType == Notes.TYPE_FOLDER || mType == Notes.TYPE_SYSTEM) {
|
|
||||||
note.put(NoteColumns.ID, mId);
|
|
||||||
note.put(NoteColumns.TYPE, mType);
|
|
||||||
note.put(NoteColumns.SNIPPET, mSnippet);
|
|
||||||
js.put(GTaskStringUtils.META_HEAD_NOTE, note);
|
|
||||||
}
|
|
||||||
|
|
||||||
return js;
|
|
||||||
} catch (JSONException e) {
|
|
||||||
Log.e(TAG, e.toString());
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setParentId(long id) {
|
|
||||||
mParentId = id;
|
|
||||||
mDiffNoteValues.put(NoteColumns.PARENT_ID, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setGtaskId(String gid) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.GTASK_ID, gid);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setSyncId(long syncId) {
|
|
||||||
mDiffNoteValues.put(NoteColumns.SYNC_ID, syncId);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void resetLocalModified() {
|
|
||||||
mDiffNoteValues.put(NoteColumns.LOCAL_MODIFIED, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public long getId() {
|
|
||||||
return mId;
|
|
||||||
}
|
|
||||||
|
|
||||||
public long getParentId() {
|
|
||||||
return mParentId;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getSnippet() {
|
|
||||||
return mSnippet;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isNoteType() {
|
|
||||||
return mType == Notes.TYPE_NOTE;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void commit(boolean validateVersion) {
|
|
||||||
if (mIsCreate) {
|
|
||||||
if (mId == INVALID_ID && mDiffNoteValues.containsKey(NoteColumns.ID)) {
|
|
||||||
mDiffNoteValues.remove(NoteColumns.ID);
|
|
||||||
}
|
|
||||||
|
|
||||||
Uri uri = mContentResolver.insert(Notes.CONTENT_NOTE_URI, mDiffNoteValues);
|
|
||||||
try {
|
|
||||||
mId = Long.valueOf(uri.getPathSegments().get(1));
|
|
||||||
} catch (NumberFormatException e) {
|
|
||||||
Log.e(TAG, "Get note id error :" + e.toString());
|
|
||||||
throw new ActionFailureException("create note failed");
|
|
||||||
}
|
|
||||||
if (mId == 0) {
|
|
||||||
throw new IllegalStateException("Create thread id failed");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mType == Notes.TYPE_NOTE) {
|
|
||||||
for (SqlData sqlData : mDataList) {
|
|
||||||
sqlData.commit(mId, false, -1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (mId <= 0 && mId != Notes.ID_ROOT_FOLDER && mId != Notes.ID_CALL_RECORD_FOLDER) {
|
|
||||||
Log.e(TAG, "No such note");
|
|
||||||
throw new IllegalStateException("Try to update note with invalid id");
|
|
||||||
}
|
|
||||||
if (mDiffNoteValues.size() > 0) {
|
|
||||||
mVersion++;
|
|
||||||
int result = 0;
|
|
||||||
if (!validateVersion) {
|
|
||||||
result = mContentResolver.update(Notes.CONTENT_NOTE_URI, mDiffNoteValues, "("
|
|
||||||
+ NoteColumns.ID + "=?)",
|
|
||||||
new String[] {
|
|
||||||
String.valueOf(mId)
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
result = mContentResolver.update(Notes.CONTENT_NOTE_URI, mDiffNoteValues, "("
|
|
||||||
+ NoteColumns.ID + "=?) AND (" + NoteColumns.VERSION + "<=?)",
|
|
||||||
new String[] {
|
|
||||||
String.valueOf(mId), String.valueOf(mVersion)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (result == 0) {
|
|
||||||
Log.w(TAG, "there is no update. maybe user updates note when syncing");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mType == Notes.TYPE_NOTE) {
|
|
||||||
for (SqlData sqlData : mDataList) {
|
|
||||||
sqlData.commit(mId, validateVersion, mVersion);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// refresh local info
|
|
||||||
loadFromCursor(mId);
|
|
||||||
if (mType == Notes.TYPE_NOTE)
|
|
||||||
loadDataContent();
|
|
||||||
|
|
||||||
mDiffNoteValues.clear();
|
|
||||||
mIsCreate = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Loading…
Reference in new issue