Compare commits
No commits in common. 'master' and 'lht_branch' have entirely different histories.
master
...
lht_branch
@ -1,232 +0,0 @@
|
|||||||
package net.micode.notes.model;
|
|
||||||
|
|
||||||
import android.content.ContentProviderOperation;//批量的更新、插入、删除数据。
|
|
||||||
import android.content.ContentProviderResult;//操作的结果
|
|
||||||
import android.content.ContentUris;//用于添加和获取Uri后面的ID
|
|
||||||
import android.content.ContentValues;//一种用来存储基本数据类型数据的存储机制
|
|
||||||
import android.content.Context;//需要用该类来弄清楚调用者的实例
|
|
||||||
import android.content.OperationApplicationException;//操作应用程序容错
|
|
||||||
import android.net.Uri;//表示待操作的数据
|
|
||||||
import android.os.RemoteException;//远程容错
|
|
||||||
import android.util.Log;//输出日志,比如说出错、警告等
|
|
||||||
|
|
||||||
public class Note {
|
|
||||||
// private ContentValues mNoteDiffValues;
|
|
||||||
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);
|
|
||||||
//ContentResolver()主要是实现外部应用对ContentProvider中的数据
|
|
||||||
//进行添加、删除、修改和查询操作
|
|
||||||
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;
|
|
||||||
}//try-catch异常处理
|
|
||||||
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);
|
|
||||||
}//设置文本数据的ID
|
|
||||||
|
|
||||||
public long getTextDataId() {
|
|
||||||
return mNoteData.mTextDataId;
|
|
||||||
}//得到文本数据的ID
|
|
||||||
|
|
||||||
public void setCallDataId(long id) {
|
|
||||||
mNoteData.setCallDataId(id);
|
|
||||||
}//设置电话号码数据的ID
|
|
||||||
|
|
||||||
public void setCallData(String key, String value) {
|
|
||||||
mNoteData.setCallData(key, value);
|
|
||||||
}//得到电话号码数据的ID
|
|
||||||
|
|
||||||
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的操作存储到数据库
|
|
||||||
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();
|
|
||||||
}//把文本数据存入DataColumns
|
|
||||||
|
|
||||||
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();
|
|
||||||
}//把电话号码数据存入DataColumns
|
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,116 +0,0 @@
|
|||||||
.micode.notes.widget;
|
|
||||||
import android.app.PendingIntent;
|
|
||||||
import android.appwidget.AppWidgetManager;
|
|
||||||
import android.appwidget.AppWidgetProvider;
|
|
||||||
import android.content.ContentValues;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.Intent;
|
|
||||||
import android.database.Cursor;
|
|
||||||
import android.util.Log;
|
|
||||||
import android.widget.RemoteViews;
|
|
||||||
|
|
||||||
import net.micode.notes.R;
|
|
||||||
import net.micode.notes.data.Notes;
|
|
||||||
import net.micode.notes.data.Notes.NoteColumns;
|
|
||||||
import net.micode.notes.tool.ResourceParser;
|
|
||||||
import net.micode.notes.ui.NoteEditActivity;
|
|
||||||
import net.micode.notes.ui.NotesListActivity;
|
|
||||||
|
|
||||||
public abstract class NoteWidgetProvider extends AppWidgetProvider {
|
|
||||||
public static final String [] PROJECTION = new String [] {
|
|
||||||
NoteColumns.ID,
|
|
||||||
NoteColumns.BG_COLOR_ID,
|
|
||||||
NoteColumns.SNIPPET
|
|
||||||
};
|
|
||||||
|
|
||||||
public static final int COLUMN_ID = 0;
|
|
||||||
public static final int COLUMN_BG_COLOR_ID = 1;
|
|
||||||
public static final int COLUMN_SNIPPET = 2;
|
|
||||||
|
|
||||||
private static final String TAG = "NoteWidgetProvider";
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onDeleted(Context context, int[] appWidgetIds) {
|
|
||||||
ContentValues values = new ContentValues();
|
|
||||||
values.put(NoteColumns.WIDGET_ID, AppWidgetManager.INVALID_APPWIDGET_ID);
|
|
||||||
for (int i = 0; i < appWidgetIds.length; i++) {
|
|
||||||
context.getContentResolver().update(Notes.CONTENT_NOTE_URI,
|
|
||||||
values,
|
|
||||||
NoteColumns.WIDGET_ID + "=?",
|
|
||||||
new String[] { String.valueOf(appWidgetIds[i])});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private Cursor getNoteWidgetInfo(Context context, int widgetId) {
|
|
||||||
return context.getContentResolver().query(Notes.CONTENT_NOTE_URI,
|
|
||||||
PROJECTION,
|
|
||||||
NoteColumns.WIDGET_ID + "=? AND " + NoteColumns.PARENT_ID + "<>?",
|
|
||||||
new String[] { String.valueOf(widgetId), String.valueOf(Notes.ID_TRASH_FOLER) },
|
|
||||||
null);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
|
|
||||||
update(context, appWidgetManager, appWidgetIds, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds,
|
|
||||||
boolean privacyMode) {
|
|
||||||
for (int i = 0; i < appWidgetIds.length; i++) {
|
|
||||||
if (appWidgetIds[i] != AppWidgetManager.INVALID_APPWIDGET_ID) {
|
|
||||||
int bgId = ResourceParser.getDefaultBgId(context);
|
|
||||||
String snippet = "";
|
|
||||||
Intent intent = new Intent(context, NoteEditActivity.class);
|
|
||||||
intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
|
|
||||||
intent.putExtra(Notes.INTENT_EXTRA_WIDGET_ID, appWidgetIds[i]);
|
|
||||||
intent.putExtra(Notes.INTENT_EXTRA_WIDGET_TYPE, getWidgetType());
|
|
||||||
|
|
||||||
Cursor c = getNoteWidgetInfo(context, appWidgetIds[i]);
|
|
||||||
if (c != null && c.moveToFirst()) {
|
|
||||||
if (c.getCount() > 1) {
|
|
||||||
Log.e(TAG, "Multiple message with same widget id:" + appWidgetIds[i]);
|
|
||||||
c.close();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
snippet = c.getString(COLUMN_SNIPPET);
|
|
||||||
bgId = c.getInt(COLUMN_BG_COLOR_ID);
|
|
||||||
intent.putExtra(Intent.EXTRA_UID, c.getLong(COLUMN_ID));
|
|
||||||
intent.setAction(Intent.ACTION_VIEW);
|
|
||||||
} else {
|
|
||||||
snippet = context.getResources().getString(R.string.widget_havenot_content);
|
|
||||||
intent.setAction(Intent.ACTION_INSERT_OR_EDIT);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (c != null) {
|
|
||||||
c.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
RemoteViews rv = new RemoteViews(context.getPackageName(), getLayoutId());
|
|
||||||
rv.setImageViewResource(R.id.widget_bg_image, getBgResourceId(bgId));
|
|
||||||
intent.putExtra(Notes.INTENT_EXTRA_BACKGROUND_ID, bgId);
|
|
||||||
/**
|
|
||||||
* Generate the pending intent to start host for the widget
|
|
||||||
*/
|
|
||||||
PendingIntent pendingIntent = null;
|
|
||||||
if (privacyMode) {
|
|
||||||
rv.setTextViewText(R.id.widget_text,
|
|
||||||
context.getString(R.string.widget_under_visit_mode));
|
|
||||||
pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], new Intent(
|
|
||||||
context, NotesListActivity.class), PendingIntent.FLAG_UPDATE_CURRENT);
|
|
||||||
} else {
|
|
||||||
rv.setTextViewText(R.id.widget_text, snippet);
|
|
||||||
pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], intent,
|
|
||||||
PendingIntent.FLAG_UPDATE_CURRENT);
|
|
||||||
}
|
|
||||||
|
|
||||||
rv.setOnClickPendingIntent(R.id.widget_text, pendingIntent);
|
|
||||||
appWidgetManager.updateAppWidget(appWidgetIds[i], rv);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected abstract int getBgResourceId(int bgId);
|
|
||||||
|
|
||||||
protected abstract int getLayoutId();
|
|
||||||
|
|
||||||
protected abstract int getWidgetType();
|
|
||||||
}
|
|
@ -1,68 +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);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,82 +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.data;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.database.Cursor;
|
|
||||||
import android.provider.ContactsContract.CommonDataKinds.Phone;
|
|
||||||
import android.provider.ContactsContract.Data;
|
|
||||||
import android.telephony.PhoneNumberUtils;
|
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import java.util.HashMap;
|
|
||||||
//change
|
|
||||||
public class Contact { //联系人
|
|
||||||
private static HashMap<String, String> sContactCache;
|
|
||||||
private static final String TAG = "Contact";
|
|
||||||
|
|
||||||
// 定义字符串CALLER_ID_SELECTION
|
|
||||||
private static final String CALLER_ID_SELECTION = "PHONE_NUMBERS_EQUAL(" + Phone.NUMBER
|
|
||||||
+ ",?) AND " + Data.MIMETYPE + "='" + Phone.CONTENT_ITEM_TYPE + "'"
|
|
||||||
+ " AND " + Data.RAW_CONTACT_ID + " IN "
|
|
||||||
+ "(SELECT raw_contact_id "
|
|
||||||
+ " FROM phone_lookup"
|
|
||||||
+ " WHERE min_match = '+')";
|
|
||||||
|
|
||||||
// 获取联系人
|
|
||||||
public static String getContact(Context context, String phoneNumber) {
|
|
||||||
if(sContactCache == null) {
|
|
||||||
sContactCache = new HashMap<String, String>();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 查找HashMap中是否已有phoneNumber信息
|
|
||||||
if(sContactCache.containsKey(phoneNumber)) {
|
|
||||||
return sContactCache.get(phoneNumber);
|
|
||||||
}
|
|
||||||
|
|
||||||
String selection = CALLER_ID_SELECTION.replace("+",
|
|
||||||
PhoneNumberUtils.toCallerIDMinMatch(phoneNumber));
|
|
||||||
// 查找数据库中phoneNumber的信息
|
|
||||||
Cursor cursor = context.getContentResolver().query(
|
|
||||||
Data.CONTENT_URI,
|
|
||||||
new String [] { Phone.DISPLAY_NAME },
|
|
||||||
selection,
|
|
||||||
new String[] { phoneNumber },
|
|
||||||
null);
|
|
||||||
|
|
||||||
// 判定查询结果
|
|
||||||
// moveToFirst()返回第一条
|
|
||||||
if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
try {
|
|
||||||
// 找到相关信息
|
|
||||||
String name = cursor.getString(0);
|
|
||||||
sContactCache.put(phoneNumber, name);
|
|
||||||
return name;
|
|
||||||
// 异常
|
|
||||||
} catch (IndexOutOfBoundsException e) {
|
|
||||||
Log.e(TAG, " Cursor get string error " + e.toString());
|
|
||||||
return null;
|
|
||||||
} finally {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
// 未找到相关信息
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "No contact matched with number:" + phoneNumber);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,82 +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.data;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.database.Cursor;
|
|
||||||
import android.provider.ContactsContract.CommonDataKinds.Phone;
|
|
||||||
import android.provider.ContactsContract.Data;
|
|
||||||
import android.telephony.PhoneNumberUtils;
|
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import java.util.HashMap;
|
|
||||||
//change
|
|
||||||
public class Contact { //联系人
|
|
||||||
private static HashMap<String, String> sContactCache;
|
|
||||||
private static final String TAG = "Contact";
|
|
||||||
|
|
||||||
// 定义字符串CALLER_ID_SELECTION
|
|
||||||
private static final String CALLER_ID_SELECTION = "PHONE_NUMBERS_EQUAL(" + Phone.NUMBER
|
|
||||||
+ ",?) AND " + Data.MIMETYPE + "='" + Phone.CONTENT_ITEM_TYPE + "'"
|
|
||||||
+ " AND " + Data.RAW_CONTACT_ID + " IN "
|
|
||||||
+ "(SELECT raw_contact_id "
|
|
||||||
+ " FROM phone_lookup"
|
|
||||||
+ " WHERE min_match = '+')";
|
|
||||||
|
|
||||||
// 获取联系人
|
|
||||||
public static String getContact(Context context, String phoneNumber) {
|
|
||||||
if(sContactCache == null) {
|
|
||||||
sContactCache = new HashMap<String, String>();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 查找HashMap中是否已有phoneNumber信息
|
|
||||||
if(sContactCache.containsKey(phoneNumber)) {
|
|
||||||
return sContactCache.get(phoneNumber);
|
|
||||||
}
|
|
||||||
|
|
||||||
String selection = CALLER_ID_SELECTION.replace("+",
|
|
||||||
PhoneNumberUtils.toCallerIDMinMatch(phoneNumber));
|
|
||||||
// 查找数据库中phoneNumber的信息
|
|
||||||
Cursor cursor = context.getContentResolver().query(
|
|
||||||
Data.CONTENT_URI,
|
|
||||||
new String [] { Phone.DISPLAY_NAME },
|
|
||||||
selection,
|
|
||||||
new String[] { phoneNumber },
|
|
||||||
null);
|
|
||||||
|
|
||||||
// 判定查询结果
|
|
||||||
// moveToFirst()返回第一条
|
|
||||||
if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
try {
|
|
||||||
// 找到相关信息
|
|
||||||
String name = cursor.getString(0);
|
|
||||||
sContactCache.put(phoneNumber, name);
|
|
||||||
return name;
|
|
||||||
// 异常
|
|
||||||
} catch (IndexOutOfBoundsException e) {
|
|
||||||
Log.e(TAG, " Cursor get string error " + e.toString());
|
|
||||||
return null;
|
|
||||||
} finally {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
// 未找到相关信息
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "No contact matched with number:" + phoneNumber);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Before Width: | Height: | Size: 43 KiB |
Before Width: | Height: | Size: 107 KiB |
Before Width: | Height: | Size: 104 KiB |
Before Width: | Height: | Size: 165 KiB |
Before Width: | Height: | Size: 139 KiB |
Before Width: | Height: | Size: 48 KiB |
Before Width: | Height: | Size: 37 KiB |
Before Width: | Height: | Size: 17 KiB |
@ -1,15 +0,0 @@
|
|||||||
*.iml
|
|
||||||
.gradle
|
|
||||||
/local.properties
|
|
||||||
/.idea/caches
|
|
||||||
/.idea/libraries
|
|
||||||
/.idea/modules.xml
|
|
||||||
/.idea/workspace.xml
|
|
||||||
/.idea/navEditor.xml
|
|
||||||
/.idea/assetWizardSettings.xml
|
|
||||||
.DS_Store
|
|
||||||
/build
|
|
||||||
/captures
|
|
||||||
.externalNativeBuild
|
|
||||||
.cxx
|
|
||||||
local.properties
|
|
@ -1 +0,0 @@
|
|||||||
/build
|
|
@ -1,66 +0,0 @@
|
|||||||
plugins {
|
|
||||||
id 'com.android.application'
|
|
||||||
id 'org.jetbrains.kotlin.android'
|
|
||||||
id 'kotlin-parcelize'
|
|
||||||
id 'kotlin-kapt'
|
|
||||||
}
|
|
||||||
|
|
||||||
android {
|
|
||||||
namespace 'net.micode.notes'
|
|
||||||
compileSdk 33
|
|
||||||
|
|
||||||
defaultConfig {
|
|
||||||
applicationId "net.micode.notes"
|
|
||||||
minSdk 24
|
|
||||||
targetSdk 33
|
|
||||||
versionCode 1
|
|
||||||
versionName "1.0"
|
|
||||||
|
|
||||||
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
|
|
||||||
|
|
||||||
javaCompileOptions {
|
|
||||||
annotationProcessorOptions {
|
|
||||||
arguments = [
|
|
||||||
"room.schemaLocation" : "$projectDir/schemas".toString(),
|
|
||||||
"room.incremental" : "true",
|
|
||||||
"room.expandProjection": "true"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
buildTypes {
|
|
||||||
release {
|
|
||||||
minifyEnabled false
|
|
||||||
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
compileOptions {
|
|
||||||
sourceCompatibility JavaVersion.VERSION_1_8
|
|
||||||
targetCompatibility JavaVersion.VERSION_1_8
|
|
||||||
}
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = '1.8'
|
|
||||||
}
|
|
||||||
buildFeatures {
|
|
||||||
viewBinding true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
|
|
||||||
implementation 'androidx.core:core-ktx:1.9.0'
|
|
||||||
implementation 'androidx.appcompat:appcompat:1.6.1'
|
|
||||||
implementation 'com.google.android.material:material:1.8.0'
|
|
||||||
implementation 'androidx.constraintlayout:constraintlayout:2.1.4'
|
|
||||||
|
|
||||||
kapt "androidx.room:room-compiler:2.4.3"
|
|
||||||
implementation 'com.github.LingChenJie:LightArchitecture:1.1.3'
|
|
||||||
implementation 'androidx.paging:paging-runtime:3.1.1'
|
|
||||||
implementation 'com.github.ihsg:PatternLocker:2.5.7'
|
|
||||||
implementation 'com.github.getActivity:TitleBar:9.2'
|
|
||||||
|
|
||||||
testImplementation 'junit:junit:4.13.2'
|
|
||||||
androidTestImplementation 'androidx.test.ext:junit:1.1.5'
|
|
||||||
androidTestImplementation 'androidx.test.espresso:espresso-core:3.5.1'
|
|
||||||
}
|
|
@ -1,21 +0,0 @@
|
|||||||
# Add project specific ProGuard rules here.
|
|
||||||
# You can control the set of applied configuration files using the
|
|
||||||
# proguardFiles setting in build.gradle.
|
|
||||||
#
|
|
||||||
# For more details, see
|
|
||||||
# http://developer.android.com/guide/developing/tools/proguard.html
|
|
||||||
|
|
||||||
# If your project uses WebView with JS, uncomment the following
|
|
||||||
# and specify the fully qualified class name to the JavaScript interface
|
|
||||||
# class:
|
|
||||||
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
|
|
||||||
# public *;
|
|
||||||
#}
|
|
||||||
|
|
||||||
# Uncomment this to preserve the line number information for
|
|
||||||
# debugging stack traces.
|
|
||||||
#-keepattributes SourceFile,LineNumberTable
|
|
||||||
|
|
||||||
# If you keep the line number information, uncomment this to
|
|
||||||
# hide the original source file name.
|
|
||||||
#-renamesourcefileattribute SourceFile
|
|
@ -1,64 +0,0 @@
|
|||||||
{
|
|
||||||
"formatVersion": 1,
|
|
||||||
"database": {
|
|
||||||
"version": 1,
|
|
||||||
"identityHash": "abc0ff5399e4a1abc3b12b90898de0da",
|
|
||||||
"entities": [
|
|
||||||
{
|
|
||||||
"tableName": "Note",
|
|
||||||
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`nId` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `create_time` INTEGER NOT NULL, `modify_time` INTEGER NOT NULL, `type` INTEGER NOT NULL)",
|
|
||||||
"fields": [
|
|
||||||
{
|
|
||||||
"fieldPath": "nId",
|
|
||||||
"columnName": "nId",
|
|
||||||
"affinity": "INTEGER",
|
|
||||||
"notNull": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldPath": "title",
|
|
||||||
"columnName": "title",
|
|
||||||
"affinity": "TEXT",
|
|
||||||
"notNull": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldPath": "content",
|
|
||||||
"columnName": "content",
|
|
||||||
"affinity": "TEXT",
|
|
||||||
"notNull": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldPath": "creteTime",
|
|
||||||
"columnName": "create_time",
|
|
||||||
"affinity": "INTEGER",
|
|
||||||
"notNull": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldPath": "modifyTime",
|
|
||||||
"columnName": "modify_time",
|
|
||||||
"affinity": "INTEGER",
|
|
||||||
"notNull": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldPath": "type",
|
|
||||||
"columnName": "type",
|
|
||||||
"affinity": "INTEGER",
|
|
||||||
"notNull": true
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"primaryKey": {
|
|
||||||
"columnNames": [
|
|
||||||
"nId"
|
|
||||||
],
|
|
||||||
"autoGenerate": true
|
|
||||||
},
|
|
||||||
"indices": [],
|
|
||||||
"foreignKeys": []
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"views": [],
|
|
||||||
"setupQueries": [
|
|
||||||
"CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)",
|
|
||||||
"INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, 'abc0ff5399e4a1abc3b12b90898de0da')"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
package com.chen.notes
|
|
||||||
|
|
||||||
import androidx.test.platform.app.InstrumentationRegistry
|
|
||||||
import androidx.test.ext.junit.runners.AndroidJUnit4
|
|
||||||
|
|
||||||
import org.junit.Test
|
|
||||||
import org.junit.runner.RunWith
|
|
||||||
|
|
||||||
import org.junit.Assert.*
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Instrumented test, which will execute on an Android device.
|
|
||||||
*
|
|
||||||
* See [testing documentation](http://d.android.com/tools/testing).
|
|
||||||
*/
|
|
||||||
@RunWith(AndroidJUnit4::class)
|
|
||||||
class ExampleInstrumentedTest {
|
|
||||||
@Test
|
|
||||||
fun useAppContext() {
|
|
||||||
// Context of the app under test.
|
|
||||||
val appContext = InstrumentationRegistry.getInstrumentation().targetContext
|
|
||||||
assertEquals("com.chen.notes", appContext.packageName)
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,8 +0,0 @@
|
|||||||
# 默认忽略的文件
|
|
||||||
/shelf/
|
|
||||||
/workspace.xml
|
|
||||||
# 数据源本地存储已忽略文件
|
|
||||||
/dataSources/
|
|
||||||
/dataSources.local.xml
|
|
||||||
# 基于编辑器的 HTTP 客户端请求
|
|
||||||
/httpRequests/
|
|
@ -1,36 +0,0 @@
|
|||||||
<component name="InspectionProjectProfileManager">
|
|
||||||
<profile version="1.0">
|
|
||||||
<option name="myName" value="Project Default" />
|
|
||||||
<inspection_tool class="JavaDoc" enabled="true" level="WARNING" enabled_by_default="true">
|
|
||||||
<option name="TOP_LEVEL_CLASS_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="INNER_CLASS_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="METHOD_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="@return@param@throws or @exception" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="FIELD_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="IGNORE_DEPRECATED" value="false" />
|
|
||||||
<option name="IGNORE_JAVADOC_PERIOD" value="true" />
|
|
||||||
<option name="IGNORE_DUPLICATED_THROWS" value="false" />
|
|
||||||
<option name="IGNORE_POINT_TO_ITSELF" value="false" />
|
|
||||||
<option name="myAdditionalJavadocTags" value="date" />
|
|
||||||
</inspection_tool>
|
|
||||||
</profile>
|
|
||||||
</component>
|
|
@ -1,6 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="ProjectRootManager">
|
|
||||||
<output url="file://$PROJECT_DIR$/out" />
|
|
||||||
</component>
|
|
||||||
</project>
|
|
@ -1,8 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="ProjectModuleManager">
|
|
||||||
<modules>
|
|
||||||
<module fileurl="file://$PROJECT_DIR$/main.iml" filepath="$PROJECT_DIR$/main.iml" />
|
|
||||||
</modules>
|
|
||||||
</component>
|
|
||||||
</project>
|
|
@ -1,6 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="VcsDirectoryMappings">
|
|
||||||
<mapping directory="$PROJECT_DIR$/../../../.." vcs="Git" />
|
|
||||||
</component>
|
|
||||||
</project>
|
|
@ -1,155 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
|
||||||
xmlns:tools="http://schemas.android.com/tools">
|
|
||||||
|
|
||||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
|
|
||||||
<uses-permission android:name="com.android.launcher.permission.INSTALL_SHORTCUT" />
|
|
||||||
<uses-permission android:name="android.permission.INTERNET" />
|
|
||||||
<uses-permission android:name="android.permission.READ_CONTACTS" />
|
|
||||||
<uses-permission android:name="android.permission.MANAGE_ACCOUNTS" />
|
|
||||||
<uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS" />
|
|
||||||
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
|
|
||||||
<uses-permission android:name="android.permission.USE_CREDENTIALS" />
|
|
||||||
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
|
|
||||||
|
|
||||||
<application
|
|
||||||
android:name=".app.App"
|
|
||||||
android:allowBackup="true"
|
|
||||||
android:dataExtractionRules="@xml/data_extraction_rules"
|
|
||||||
android:fullBackupContent="@xml/backup_rules"
|
|
||||||
android:icon="@mipmap/ic_launcher"
|
|
||||||
android:label="@string/app_name"
|
|
||||||
android:supportsRtl="true"
|
|
||||||
android:theme="@style/Theme.Notes"
|
|
||||||
tools:targetApi="31">
|
|
||||||
<!--手势密码入口-->
|
|
||||||
<activity
|
|
||||||
android:name="net.micode.notes.ui.page.MainActivity"
|
|
||||||
android:exported="true">
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.intent.action.MAIN" />
|
|
||||||
|
|
||||||
<category android:name="android.intent.category.LAUNCHER" />
|
|
||||||
</intent-filter>
|
|
||||||
</activity>
|
|
||||||
<!--安全文件夹入口-->
|
|
||||||
<activity android:name="net.micode.notes.ui.page.mvi.NotesActivity" />
|
|
||||||
|
|
||||||
<activity
|
|
||||||
android:name="net.micode.notes.ui.NotesListActivity"
|
|
||||||
android:configChanges="keyboardHidden|orientation|screenSize"
|
|
||||||
android:exported="true"
|
|
||||||
android:label="@string/app_name"
|
|
||||||
android:launchMode="singleTop"
|
|
||||||
android:theme="@style/NoteTheme"
|
|
||||||
android:uiOptions="splitActionBarWhenNarrow"
|
|
||||||
android:windowSoftInputMode="adjustPan">
|
|
||||||
|
|
||||||
<!-- <intent-filter>-->
|
|
||||||
<!-- <action android:name="android.intent.action.MAIN" />-->
|
|
||||||
<!-- <category android:name="android.intent.category.LAUNCHER" />-->
|
|
||||||
<!-- </intent-filter>-->
|
|
||||||
</activity>
|
|
||||||
|
|
||||||
|
|
||||||
<activity
|
|
||||||
android:name="net.micode.notes.ui.NoteEditActivity"
|
|
||||||
android:configChanges="keyboardHidden|orientation|screenSize"
|
|
||||||
android:exported="true"
|
|
||||||
android:launchMode="singleTop"
|
|
||||||
android:theme="@style/NoteTheme">
|
|
||||||
|
|
||||||
<!-- <intent-filter>-->
|
|
||||||
<!-- <action android:name="android.intent.action.VIEW" />-->
|
|
||||||
<!-- <category android:name="android.intent.category.DEFAULT" />-->
|
|
||||||
<!-- <data android:mimeType="vnd.android.cursor.item/text_note" />-->
|
|
||||||
<!-- <data android:mimeType="vnd.android.cursor.item/call_note" />-->
|
|
||||||
<!-- </intent-filter>-->
|
|
||||||
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.intent.action.INSERT_OR_EDIT" />
|
|
||||||
<category android:name="android.intent.category.DEFAULT" />
|
|
||||||
|
|
||||||
<data android:mimeType="vnd.android.cursor.item/text_note" />
|
|
||||||
<data android:mimeType="vnd.android.cursor.item/call_note" />
|
|
||||||
</intent-filter>
|
|
||||||
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.intent.action.SEARCH" />
|
|
||||||
<category android:name="android.intent.category.DEFAULT" />
|
|
||||||
</intent-filter>
|
|
||||||
|
|
||||||
<meta-data
|
|
||||||
android:name="android.app.searchable"
|
|
||||||
android:resource="@xml/searchable" />
|
|
||||||
</activity>
|
|
||||||
|
|
||||||
<provider
|
|
||||||
android:name="net.micode.notes.data.NotesProvider"
|
|
||||||
android:authorities="micode_notes"
|
|
||||||
android:multiprocess="true" />
|
|
||||||
|
|
||||||
<receiver
|
|
||||||
android:name="net.micode.notes.widget.NoteWidgetProvider_2x"
|
|
||||||
android:exported="true"
|
|
||||||
android:label="@string/app_widget2x2">
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
|
|
||||||
<action android:name="android.appwidget.action.APPWIDGET_DELETED" />
|
|
||||||
<action android:name="android.intent.action.PRIVACY_MODE_CHANGED" />
|
|
||||||
</intent-filter>
|
|
||||||
|
|
||||||
<meta-data
|
|
||||||
android:name="android.appwidget.provider"
|
|
||||||
android:resource="@xml/widget_2x_info" />
|
|
||||||
</receiver>
|
|
||||||
<receiver
|
|
||||||
android:name="net.micode.notes.widget.NoteWidgetProvider_4x"
|
|
||||||
android:exported="true"
|
|
||||||
android:label="@string/app_widget4x4">
|
|
||||||
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
|
|
||||||
<action android:name="android.appwidget.action.APPWIDGET_DELETED" />
|
|
||||||
<action android:name="android.intent.action.PRIVACY_MODE_CHANGED" />
|
|
||||||
</intent-filter>
|
|
||||||
|
|
||||||
<meta-data
|
|
||||||
android:name="android.appwidget.provider"
|
|
||||||
android:resource="@xml/widget_4x_info" />
|
|
||||||
</receiver>
|
|
||||||
|
|
||||||
<receiver
|
|
||||||
android:name="net.micode.notes.ui.AlarmInitReceiver"
|
|
||||||
android:exported="true">
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.intent.action.BOOT_COMPLETED" />
|
|
||||||
</intent-filter>
|
|
||||||
</receiver>
|
|
||||||
|
|
||||||
<receiver
|
|
||||||
android:name="net.micode.notes.ui.AlarmReceiver"
|
|
||||||
android:process=":remote"></receiver>
|
|
||||||
|
|
||||||
<activity
|
|
||||||
android:name="net.micode.notes.ui.AlarmAlertActivity"
|
|
||||||
android:label="@string/app_name"
|
|
||||||
android:launchMode="singleInstance"
|
|
||||||
android:theme="@android:style/Theme.Holo.Wallpaper.NoTitleBar"></activity>
|
|
||||||
|
|
||||||
<activity
|
|
||||||
android:name="net.micode.notes.ui.NotesPreferenceActivity"
|
|
||||||
android:label="@string/preferences_title"
|
|
||||||
android:launchMode="singleTop"
|
|
||||||
android:theme="@android:style/Theme.Holo.Light"></activity>
|
|
||||||
|
|
||||||
<service
|
|
||||||
android:name="net.micode.notes.gtask.remote.GTaskSyncService"
|
|
||||||
android:exported="false"></service>
|
|
||||||
|
|
||||||
<meta-data
|
|
||||||
android:name="android.app.default_searchable"
|
|
||||||
android:value=".ui.NoteEditActivity" />
|
|
||||||
</application>
|
|
||||||
|
|
||||||
</manifest>
|
|
@ -1,8 +0,0 @@
|
|||||||
# 默认忽略的文件
|
|
||||||
/shelf/
|
|
||||||
/workspace.xml
|
|
||||||
# 数据源本地存储已忽略文件
|
|
||||||
/dataSources/
|
|
||||||
/dataSources.local.xml
|
|
||||||
# 基于编辑器的 HTTP 客户端请求
|
|
||||||
/httpRequests/
|
|
@ -1,36 +0,0 @@
|
|||||||
<component name="InspectionProjectProfileManager">
|
|
||||||
<profile version="1.0">
|
|
||||||
<option name="myName" value="Project Default" />
|
|
||||||
<inspection_tool class="JavaDoc" enabled="true" level="WARNING" enabled_by_default="true">
|
|
||||||
<option name="TOP_LEVEL_CLASS_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="INNER_CLASS_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="METHOD_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="@return@param@throws or @exception" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="FIELD_OPTIONS">
|
|
||||||
<value>
|
|
||||||
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
|
|
||||||
<option name="REQUIRED_TAGS" value="" />
|
|
||||||
</value>
|
|
||||||
</option>
|
|
||||||
<option name="IGNORE_DEPRECATED" value="false" />
|
|
||||||
<option name="IGNORE_JAVADOC_PERIOD" value="true" />
|
|
||||||
<option name="IGNORE_DUPLICATED_THROWS" value="false" />
|
|
||||||
<option name="IGNORE_POINT_TO_ITSELF" value="false" />
|
|
||||||
<option name="myAdditionalJavadocTags" value="date" />
|
|
||||||
</inspection_tool>
|
|
||||||
</profile>
|
|
||||||
</component>
|
|
@ -1,6 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="ProjectRootManager">
|
|
||||||
<output url="file://$PROJECT_DIR$/out" />
|
|
||||||
</component>
|
|
||||||
</project>
|
|
@ -1,8 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="ProjectModuleManager">
|
|
||||||
<modules>
|
|
||||||
<module fileurl="file://$PROJECT_DIR$/main.iml" filepath="$PROJECT_DIR$/main.iml" />
|
|
||||||
</modules>
|
|
||||||
</component>
|
|
||||||
</project>
|
|
@ -1,6 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="VcsDirectoryMappings">
|
|
||||||
<mapping directory="$PROJECT_DIR$/../../../../.." vcs="Git" />
|
|
||||||
</component>
|
|
||||||
</project>
|
|
@ -1,14 +0,0 @@
|
|||||||
package net.micode.notes.app
|
|
||||||
|
|
||||||
import com.android.architecture.app.BaseApplication
|
|
||||||
import com.android.architecture.helper.Logger
|
|
||||||
import net.micode.notes.data.Constant
|
|
||||||
|
|
||||||
class App : BaseApplication() {
|
|
||||||
|
|
||||||
override fun onCreate() {
|
|
||||||
super.onCreate()
|
|
||||||
Logger.e(Constant.TAG, "onCreate()")
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,76 +0,0 @@
|
|||||||
package net.micode.notes.app
|
|
||||||
|
|
||||||
import android.os.Bundle
|
|
||||||
import android.view.ViewGroup
|
|
||||||
import com.android.architecture.ui.page.BaseActivity
|
|
||||||
import com.android.architecture.ui.page.BaseDialog
|
|
||||||
import com.android.architecture.ui.widget.layout.TitleView
|
|
||||||
import com.gyf.immersionbar.ImmersionBar
|
|
||||||
|
|
||||||
abstract class AppActivity : BaseActivity() {
|
|
||||||
|
|
||||||
override fun onCreate(savedInstanceState: Bundle?) {
|
|
||||||
super.onCreate(savedInstanceState)
|
|
||||||
if (isStatusBarEnabled()) {
|
|
||||||
getStatusBarConfig().init()
|
|
||||||
getTitleView()?.let {
|
|
||||||
ImmersionBar.setTitleBar(this, it)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 是否使用沉浸式状态栏
|
|
||||||
*/
|
|
||||||
open fun isStatusBarEnabled(): Boolean {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 状态栏字体深色模式
|
|
||||||
*/
|
|
||||||
open fun isStatusBarDarkFont(): Boolean {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
private var mLoadingDialog: BaseDialog? = null
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 获取状态栏沉浸的配置对象
|
|
||||||
*/
|
|
||||||
private fun getStatusBarConfig(): ImmersionBar {
|
|
||||||
return ImmersionBar.with(this)
|
|
||||||
.statusBarDarkFont(isStatusBarDarkFont())
|
|
||||||
.autoDarkModeEnable(true, 0.2f)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 获取标题栏
|
|
||||||
*/
|
|
||||||
private fun getTitleView(): TitleView? {
|
|
||||||
return obtainTitleBar(contentView)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 递归获取 ViewGroup 中的 TitleBar 对象
|
|
||||||
*/
|
|
||||||
private fun obtainTitleBar(group: ViewGroup?): TitleView? {
|
|
||||||
if (group == null) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
for (i in 0 until group.childCount) {
|
|
||||||
val view = group.getChildAt(i)
|
|
||||||
if (view is TitleView) {
|
|
||||||
return view
|
|
||||||
}
|
|
||||||
if (view is ViewGroup) {
|
|
||||||
val titleBar: TitleView? = obtainTitleBar(view)
|
|
||||||
if (titleBar != null) {
|
|
||||||
return titleBar
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|