|
|
@ -14,119 +14,156 @@
|
|
|
|
* limitations under the License.
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
package net.micode.notes.widget;
|
|
|
|
package net.micode.notes.widget; // 定义包名,表示这个类所在的包
|
|
|
|
import android.app.PendingIntent;
|
|
|
|
|
|
|
|
import android.appwidget.AppWidgetManager;
|
|
|
|
import android.app.PendingIntent; // 导入PendingIntent类,用于创建延迟的Intent
|
|
|
|
import android.appwidget.AppWidgetProvider;
|
|
|
|
import android.appwidget.AppWidgetManager; // 导入AppWidgetManager类,用于管理小部件
|
|
|
|
import android.content.ContentValues;
|
|
|
|
import android.appwidget.AppWidgetProvider; // 导入AppWidgetProvider类,用于定义小部件的行为
|
|
|
|
import android.content.Context;
|
|
|
|
import android.content.ContentValues; // 导入ContentValues类,用于存储要插入或更新的值
|
|
|
|
import android.content.Intent;
|
|
|
|
import android.content.Context; // 导入Context类,用于访问应用程序环境
|
|
|
|
import android.database.Cursor;
|
|
|
|
import android.content.Intent; // 导入Intent类,用于在不同组件之间进行通信
|
|
|
|
import android.util.Log;
|
|
|
|
import android.database.Cursor; // 导入Cursor类,用于查询返回的数据集
|
|
|
|
import android.widget.RemoteViews;
|
|
|
|
import android.util.Log; // 导入Log类,用于日志输出
|
|
|
|
|
|
|
|
import android.widget.RemoteViews; // 导入RemoteViews类,用于定义小部件的布局和行为
|
|
|
|
import net.micode.notes.R;
|
|
|
|
|
|
|
|
import net.micode.notes.data.Notes;
|
|
|
|
import net.micode.notes.R; // 导入R类,用于访问应用程序的资源文件
|
|
|
|
import net.micode.notes.data.Notes.NoteColumns;
|
|
|
|
import net.micode.notes.data.Notes; // 导入Notes类,包含便签相关的数据操作
|
|
|
|
import net.micode.notes.tool.ResourceParser;
|
|
|
|
import net.micode.notes.data.Notes.NoteColumns; // 导入NoteColumns类,包含便签数据表的列名
|
|
|
|
import net.micode.notes.ui.NoteEditActivity;
|
|
|
|
import net.micode.notes.tool.ResourceParser; // 导入ResourceParser类,用于解析资源
|
|
|
|
import net.micode.notes.ui.NotesListActivity;
|
|
|
|
import net.micode.notes.ui.NoteEditActivity; // 导入NoteEditActivity类,用于编辑便签
|
|
|
|
|
|
|
|
import net.micode.notes.ui.NotesListActivity; // 导入NotesListActivity类,用于显示便签列表
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 定义一个抽象的NoteWidgetProvider类,继承自AppWidgetProvider
|
|
|
|
|
|
|
|
// 这个类负责处理便签桌面小部件的通用逻辑
|
|
|
|
public abstract class NoteWidgetProvider extends AppWidgetProvider {
|
|
|
|
public abstract class NoteWidgetProvider extends AppWidgetProvider {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 定义查询便签时使用的列投影
|
|
|
|
public static final String [] PROJECTION = new String [] {
|
|
|
|
public static final String [] PROJECTION = new String [] {
|
|
|
|
NoteColumns.ID,
|
|
|
|
NoteColumns.ID, // 便签ID
|
|
|
|
NoteColumns.BG_COLOR_ID,
|
|
|
|
NoteColumns.BG_COLOR_ID, // 便签背景颜色ID
|
|
|
|
NoteColumns.SNIPPET
|
|
|
|
NoteColumns.SNIPPET // 便签内容摘要
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
public static final int COLUMN_ID = 0;
|
|
|
|
// 定义列索引常量,便于通过索引访问查询结果中的数据
|
|
|
|
public static final int COLUMN_BG_COLOR_ID = 1;
|
|
|
|
public static final int COLUMN_ID = 0; // 便签ID的列索引
|
|
|
|
public static final int COLUMN_SNIPPET = 2;
|
|
|
|
public static final int COLUMN_BG_COLOR_ID = 1; // 便签背景颜色ID的列索引
|
|
|
|
|
|
|
|
public static final int COLUMN_SNIPPET = 2; // 便签内容摘要的列索引
|
|
|
|
|
|
|
|
|
|
|
|
private static final String TAG = "NoteWidgetProvider";
|
|
|
|
// 定义日志标签
|
|
|
|
|
|
|
|
private static final String TAG = "NoteWidgetProvider"; // 日志标签,便于在日志中识别来自这个类的日志信息
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 当小部件被删除时调用的方法
|
|
|
|
|
|
|
|
// @param context 应用上下文
|
|
|
|
|
|
|
|
// @param appWidgetIds 被删除的小部件ID数组
|
|
|
|
@Override
|
|
|
|
@Override
|
|
|
|
public void onDeleted(Context context, int[] appWidgetIds) {
|
|
|
|
public void onDeleted(Context context, int[] appWidgetIds) {
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
ContentValues values = new ContentValues(); // 创建一个ContentValues对象,用于存储要更新的值
|
|
|
|
values.put(NoteColumns.WIDGET_ID, AppWidgetManager.INVALID_APPWIDGET_ID);
|
|
|
|
values.put(NoteColumns.WIDGET_ID, AppWidgetManager.INVALID_APPWIDGET_ID); // 设置便签的WIDGET_ID为无效ID,表示该便签不再关联任何小部件
|
|
|
|
for (int i = 0; i < appWidgetIds.length; i++) {
|
|
|
|
for (int i = 0; i < appWidgetIds.length; i++) { // 遍历被删除的小部件ID数组
|
|
|
|
context.getContentResolver().update(Notes.CONTENT_NOTE_URI,
|
|
|
|
// 更新数据库中便签的WIDGET_ID为无效ID
|
|
|
|
values,
|
|
|
|
context.getContentResolver().update(Notes.CONTENT_NOTE_URI, // 数据表的URI
|
|
|
|
NoteColumns.WIDGET_ID + "=?",
|
|
|
|
values, // 要更新的值
|
|
|
|
new String[] { String.valueOf(appWidgetIds[i])});
|
|
|
|
NoteColumns.WIDGET_ID + "=?", // 查询条件,WIDGET_ID等于特定值
|
|
|
|
|
|
|
|
new String[] { String.valueOf(appWidgetIds[i]) }); // 查询条件的值,即当前小部件ID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 获取指定小部件ID的便签信息
|
|
|
|
|
|
|
|
// @param context 应用上下文
|
|
|
|
|
|
|
|
// @param widgetId 小部件ID
|
|
|
|
|
|
|
|
// @return 查询结果的Cursor对象
|
|
|
|
private Cursor getNoteWidgetInfo(Context context, int widgetId) {
|
|
|
|
private Cursor getNoteWidgetInfo(Context context, int widgetId) {
|
|
|
|
return context.getContentResolver().query(Notes.CONTENT_NOTE_URI,
|
|
|
|
// 查询数据库中WIDGET_ID等于特定值且PARENT_ID不等于废纸篓ID的便签信息
|
|
|
|
PROJECTION,
|
|
|
|
return context.getContentResolver().query(Notes.CONTENT_NOTE_URI, // 数据表的URI
|
|
|
|
NoteColumns.WIDGET_ID + "=? AND " + NoteColumns.PARENT_ID + "<>?",
|
|
|
|
PROJECTION, // 查询的列
|
|
|
|
new String[] { String.valueOf(widgetId), String.valueOf(Notes.ID_TRASH_FOLER) },
|
|
|
|
NoteColumns.WIDGET_ID + "=? AND " + NoteColumns.PARENT_ID + "!=?", // 查询条件
|
|
|
|
null);
|
|
|
|
new String[] { String.valueOf(widgetId), String.valueOf(Notes.ID_TRASH_FOLER) }, // 查询条件的值
|
|
|
|
|
|
|
|
null); // 排序方式,这里为null表示不排序
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 更新小部件的方法,调用另一个带有privacyMode参数的update方法
|
|
|
|
|
|
|
|
// @param context 应用上下文
|
|
|
|
|
|
|
|
// @param appWidgetManager 小部件管理器
|
|
|
|
|
|
|
|
// @param appWidgetIds 需要更新的小部件ID数组
|
|
|
|
protected void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
|
|
|
|
protected void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
|
|
|
|
update(context, appWidgetManager, appWidgetIds, false);
|
|
|
|
update(context, appWidgetManager, appWidgetIds, false); // 调用另一个update方法,privacyMode默认为false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 更新小部件的方法,处理小部件的更新逻辑
|
|
|
|
|
|
|
|
// @param context 应用上下文
|
|
|
|
|
|
|
|
// @param appWidgetManager 小部件管理器
|
|
|
|
|
|
|
|
// @param appWidgetIds 需要更新的小部件ID数组
|
|
|
|
|
|
|
|
// @param privacyMode 是否处于隐私模式
|
|
|
|
private void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds,
|
|
|
|
private void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds,
|
|
|
|
boolean privacyMode) {
|
|
|
|
boolean privacyMode) {
|
|
|
|
for (int i = 0; i < appWidgetIds.length; i++) {
|
|
|
|
for (int i = 0; i < appWidgetIds.length; i++) { // 遍历需要更新的小部件ID数组
|
|
|
|
if (appWidgetIds[i] != AppWidgetManager.INVALID_APPWIDGET_ID) {
|
|
|
|
if (appWidgetIds[i] != AppWidgetManager.INVALID_APPWIDGET_ID) { // 检查当前小部件ID是否有效
|
|
|
|
int bgId = ResourceParser.getDefaultBgId(context);
|
|
|
|
int bgId = ResourceParser.getDefaultBgId(context); // 获取默认背景ID
|
|
|
|
String snippet = "";
|
|
|
|
String snippet = ""; // 初始化便签内容摘要为空字符串
|
|
|
|
Intent intent = new Intent(context, NoteEditActivity.class);
|
|
|
|
Intent intent = new Intent(context, NoteEditActivity.class); // 创建一个Intent对象,指向NoteEditActivity
|
|
|
|
intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
|
|
|
|
intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP); // 设置Intent标志,表示如果目标Activity已经在任务栈顶,则不新建实例
|
|
|
|
intent.putExtra(Notes.INTENT_EXTRA_WIDGET_ID, appWidgetIds[i]);
|
|
|
|
intent.putExtra(Notes.INTENT_EXTRA_WIDGET_ID, appWidgetIds[i]); // 将小部件ID添加到Intent中
|
|
|
|
intent.putExtra(Notes.INTENT_EXTRA_WIDGET_TYPE, getWidgetType());
|
|
|
|
intent.putExtra(Notes.INTENT_EXTRA_WIDGET_TYPE, getWidgetType()); // 将小部件类型添加到Intent中
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 查询指定小部件ID的便签信息
|
|
|
|
Cursor c = getNoteWidgetInfo(context, appWidgetIds[i]);
|
|
|
|
Cursor c = getNoteWidgetInfo(context, appWidgetIds[i]);
|
|
|
|
if (c != null && c.moveToFirst()) {
|
|
|
|
if (c != null && c.moveToFirst()) { // 检查查询结果是否有效并且有数据
|
|
|
|
if (c.getCount() > 1) {
|
|
|
|
if (c.getCount() > 1) { // 检查查询结果的数量是否大于1
|
|
|
|
Log.e(TAG, "Multiple message with same widget id:" + appWidgetIds[i]);
|
|
|
|
Log.e(TAG, "Multiple message with same widget id:" + appWidgetIds[i]); // 记录错误日志,表示有多个便签关联了同一个小部件ID
|
|
|
|
c.close();
|
|
|
|
c.close(); // 关闭Cursor
|
|
|
|
return;
|
|
|
|
return; // 返回,不继续执行
|
|
|
|
}
|
|
|
|
}
|
|
|
|
snippet = c.getString(COLUMN_SNIPPET);
|
|
|
|
snippet = c.getString(COLUMN_SNIPPET); // 获取便签内容摘要
|
|
|
|
bgId = c.getInt(COLUMN_BG_COLOR_ID);
|
|
|
|
bgId = c.getInt(COLUMN_BG_COLOR_ID); // 获取便签背景颜色ID
|
|
|
|
intent.putExtra(Intent.EXTRA_UID, c.getLong(COLUMN_ID));
|
|
|
|
intent.putExtra(Intent.EXTRA_UID, c.getLong(COLUMN_ID)); // 将便签ID添加到Intent中
|
|
|
|
intent.setAction(Intent.ACTION_VIEW);
|
|
|
|
intent.setAction(Intent.ACTION_VIEW); // 设置Intent动作,表示查看便签
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
snippet = context.getResources().getString(R.string.widget_havenot_content);
|
|
|
|
snippet = context.getResources().getString(R.string.widget_havenot_content); // 设置便签内容摘要为“没有内容”
|
|
|
|
intent.setAction(Intent.ACTION_INSERT_OR_EDIT);
|
|
|
|
intent.setAction(Intent.ACTION_INSERT_OR_EDIT); // 设置Intent动作,表示插入或编辑便签
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (c != null) {
|
|
|
|
if (c != null) { // 检查Cursor是否有效
|
|
|
|
c.close();
|
|
|
|
c.close(); // 关闭Cursor
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
RemoteViews rv = new RemoteViews(context.getPackageName(), getLayoutId());
|
|
|
|
// 创建一个RemoteViews对象,用于定义小部件的布局和行为
|
|
|
|
rv.setImageViewResource(R.id.widget_bg_image, getBgResourceId(bgId));
|
|
|
|
RemoteViews rv = new RemoteViews(context.getPackageName(), getLayoutId()); // 获取小部件的布局资源ID
|
|
|
|
intent.putExtra(Notes.INTENT_EXTRA_BACKGROUND_ID, bgId);
|
|
|
|
rv.setImageViewResource(R.id.widget_bg_image, getBgResourceId(bgId)); // 设置小部件背景图像资源ID
|
|
|
|
|
|
|
|
intent.putExtra(Notes.INTENT_EXTRA_BACKGROUND_ID, bgId); // 将背景ID添加到Intent中
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Generate the pending intent to start host for the widget
|
|
|
|
* 生成用于启动宿主Activity的PendingIntent
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
PendingIntent pendingIntent = null;
|
|
|
|
PendingIntent pendingIntent = null; // 初始化PendingIntent为null
|
|
|
|
if (privacyMode) {
|
|
|
|
if (privacyMode) { // 如果处于隐私模式
|
|
|
|
rv.setTextViewText(R.id.widget_text,
|
|
|
|
rv.setTextViewText(R.id.widget_text, // 设置小部件文本视图的内容
|
|
|
|
context.getString(R.string.widget_under_visit_mode));
|
|
|
|
context.getString(R.string.widget_under_visit_mode)); // 设置内容为“正在访问模式”
|
|
|
|
|
|
|
|
// 创建一个PendingIntent,指向NotesListActivity
|
|
|
|
pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], new Intent(
|
|
|
|
pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], new Intent(
|
|
|
|
context, NotesListActivity.class), PendingIntent.FLAG_UPDATE_CURRENT);
|
|
|
|
context, NotesListActivity.class), PendingIntent.FLAG_UPDATE_CURRENT);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
rv.setTextViewText(R.id.widget_text, snippet);
|
|
|
|
rv.setTextViewText(R.id.widget_text, snippet); // 设置小部件文本视图的内容为便签摘要
|
|
|
|
|
|
|
|
// 创建一个PendingIntent,指向NoteEditActivity
|
|
|
|
pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], intent,
|
|
|
|
pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], intent,
|
|
|
|
PendingIntent.FLAG_UPDATE_CURRENT);
|
|
|
|
PendingIntent.FLAG_UPDATE_CURRENT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
rv.setOnClickPendingIntent(R.id.widget_text, pendingIntent);
|
|
|
|
rv.setOnClickPendingIntent(R.id.widget_text, pendingIntent); // 设置小部件文本视图的点击事件为PendingIntent
|
|
|
|
appWidgetManager.updateAppWidget(appWidgetIds[i], rv);
|
|
|
|
appWidgetManager.updateAppWidget(appWidgetIds[i], rv); // 更新指定小部件的布局
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 获取便签背景资源ID的方法,抽象方法,由子类实现
|
|
|
|
|
|
|
|
// @param bgId 背景样式ID
|
|
|
|
|
|
|
|
// @return 对应的背景资源ID
|
|
|
|
protected abstract int getBgResourceId(int bgId);
|
|
|
|
protected abstract int getBgResourceId(int bgId);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 获取小部件布局资源ID的方法,抽象方法,由子类实现
|
|
|
|
|
|
|
|
// @return 小部件布局资源ID
|
|
|
|
protected abstract int getLayoutId();
|
|
|
|
protected abstract int getLayoutId();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 获取小部件类型的方法,抽象方法,由子类实现
|
|
|
|
|
|
|
|
// @return 小部件类型ID
|
|
|
|
protected abstract int getWidgetType();
|
|
|
|
protected abstract int getWidgetType();
|
|
|
|
}
|
|
|
|
}
|
|
|
|