diff --git a/doc/报告.docx b/doc/报告.docx new file mode 100644 index 0000000..e69de29 diff --git a/doc/赵鑫 泛读报告第一节.docx b/doc/赵鑫 泛读报告第一节.docx new file mode 100644 index 0000000..fa1a0d2 Binary files /dev/null and b/doc/赵鑫 泛读报告第一节.docx differ diff --git a/src/Notes-master/src/net/micode/notes/widget/NoteWidgetProvider.java b/src/Notes-master/src/net/micode/notes/widget/NoteWidgetProvider.java index ec6f819..2b1ba2a 100644 --- a/src/Notes-master/src/net/micode/notes/widget/NoteWidgetProvider.java +++ b/src/Notes-master/src/net/micode/notes/widget/NoteWidgetProvider.java @@ -14,119 +14,222 @@ * limitations under the License. */ +// 包声明:归属小米便签桌面小部件模块,定义所有尺寸便签小部件的通用基类 package net.micode.notes.widget; + +// 导入安卓延迟意图类:用于小部件点击事件的异步意图触发 import android.app.PendingIntent; +// 导入安卓小部件管理器类:管理小部件的生命周期(更新、删除、创建) import android.appwidget.AppWidgetManager; +// 导入安卓小部件提供者基类:所有桌面小部件的系统基类 import android.appwidget.AppWidgetProvider; +// 导入安卓内容值类:用于封装ContentProvider的更新数据 import android.content.ContentValues; +// 导入安卓上下文类:提供应用运行环境(资源访问、ContentResolver获取等) import android.content.Context; +// 导入安卓意图类:用于组件间通信(小部件点击跳转页面) import android.content.Intent; +// 导入安卓数据库游标类:存储数据库查询结果 import android.database.Cursor; +// 导入安卓日志类:输出小部件相关日志(异常/调试) import android.util.Log; +// 导入安卓远程视图类:用于渲染桌面小部件的UI(小部件运行在桌面进程,需远程视图) import android.widget.RemoteViews; +// 导入小米便签资源类:引用布局、字符串、图片等资源 import net.micode.notes.R; +// 导入便签数据常量类:定义便签URI、字段、意图参数等核心常量 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; +/** + * 便签桌面小部件抽象基类 + * 继承自安卓系统AppWidgetProvider,封装所有尺寸便签小部件的通用核心逻辑: + * 1. 小部件删除时清理关联的便签数据; + * 2. 小部件更新时的通用UI渲染、数据查询、点击事件绑定; + * 3. 定义抽象方法,由子类(2x/4x)实现尺寸相关的布局、背景、类型适配。 + */ public abstract class NoteWidgetProvider extends AppWidgetProvider { + /** + * 数据库查询投影数组:指定查询便签的核心字段,减少查询冗余 + * 包含:便签ID、背景色ID、便签摘要(用于小部件展示) + */ public static final String [] PROJECTION = new String [] { - NoteColumns.ID, - NoteColumns.BG_COLOR_ID, - NoteColumns.SNIPPET + NoteColumns.ID, // 便签唯一标识 + NoteColumns.BG_COLOR_ID, // 便签背景色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; + // 投影数组对应的列索引常量,简化Cursor取值 + public static final int COLUMN_ID = 0; // 便签ID列索引 + public static final int COLUMN_BG_COLOR_ID = 1; // 背景色ID列索引 + public static final int COLUMN_SNIPPET = 2; // 摘要列索引 + // 日志标签:用于小部件相关日志输出,便于问题定位 private static final String TAG = "NoteWidgetProvider"; + /** + * 重写小部件删除回调方法 + * 当用户删除便签小部件时触发,核心逻辑:将关联便签的WIDGET_ID置为无效,清理关联关系 + * @param context 应用上下文 + * @param appWidgetIds 被删除的小部件ID数组 + */ @Override public void onDeleted(Context context, int[] appWidgetIds) { + // 构建ContentValues:将WIDGET_ID设为无效值(INVALID_APPWIDGET_ID) ContentValues values = new ContentValues(); values.put(NoteColumns.WIDGET_ID, AppWidgetManager.INVALID_APPWIDGET_ID); + + // 遍历所有被删除的小部件ID,更新关联便签的WIDGET_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])}); + context.getContentResolver().update(Notes.CONTENT_NOTE_URI, // 便签ContentProvider的URI + values, // 要更新的字段和值 + NoteColumns.WIDGET_ID + "=?", // 更新条件:WIDGET_ID匹配当前删除的ID + new String[] { String.valueOf(appWidgetIds[i])}); // 条件参数(防止SQL注入) } } + /** + * 私有方法:根据小部件ID查询关联的便签信息 + * 过滤条件:关联当前widgetId + 非回收站便签(PARENT_ID≠回收站ID) + * @param context 应用上下文,用于获取ContentResolver + * @param widgetId 目标小部件ID + * @return Cursor:包含匹配的便签信息(ID、背景色、摘要),无匹配则返回null + */ 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); + return context.getContentResolver().query(Notes.CONTENT_NOTE_URI, // 便签查询URI + PROJECTION, // 查询的字段投影 + NoteColumns.WIDGET_ID + "=? AND " + NoteColumns.PARENT_ID + "<>?", // 查询条件 + new String[] { String.valueOf(widgetId), String.valueOf(Notes.ID_TRASH_FOLER) }, // 条件参数 + null); // 排序规则(无) } + /** + * 对外暴露的小部件更新方法(重载) + * 默认以非隐私模式更新小部件,调用带privacyMode的核心更新方法 + * @param context 应用上下文 + * @param appWidgetManager 小部件管理器 + * @param appWidgetIds 需要更新的小部件ID数组 + */ protected void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { update(context, appWidgetManager, appWidgetIds, false); } + /** + * 核心私有更新方法:实现小部件UI渲染、数据绑定、点击事件配置的通用逻辑 + * @param context 应用上下文 + * @param appWidgetManager 小部件管理器,用于更新小部件UI + * @param appWidgetIds 需要更新的小部件ID数组 + * @param privacyMode 是否为隐私模式(隐私模式下隐藏内容,跳转列表页) + */ private void update(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds, boolean privacyMode) { + // 遍历所有需要更新的小部件ID,逐个处理 for (int i = 0; i < appWidgetIds.length; i++) { + // 过滤无效的小部件ID(INVALID_APPWIDGET_ID) if (appWidgetIds[i] != AppWidgetManager.INVALID_APPWIDGET_ID) { + // 初始化默认背景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()); + intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP); // 防止重复创建Activity + intent.putExtra(Notes.INTENT_EXTRA_WIDGET_ID, appWidgetIds[i]); // 携带小部件ID参数 + intent.putExtra(Notes.INTENT_EXTRA_WIDGET_TYPE, getWidgetType()); // 携带小部件类型(子类实现) + // 查询当前小部件关联的便签信息 Cursor c = getNoteWidgetInfo(context, appWidgetIds[i]); if (c != null && c.moveToFirst()) { + // 异常日志:同一个widgetId关联多个便签(数据异常) if (c.getCount() > 1) { Log.e(TAG, "Multiple message with same widget id:" + appWidgetIds[i]); c.close(); return; } + // 从Cursor中获取便签摘要和背景色ID snippet = c.getString(COLUMN_SNIPPET); bgId = c.getInt(COLUMN_BG_COLOR_ID); + // 携带便签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); } + // 关闭Cursor,释放数据库资源 if (c != null) { c.close(); } + // 创建远程视图:加载子类定义的布局(2x/4x) RemoteViews rv = new RemoteViews(context.getPackageName(), getLayoutId()); + // 设置小部件背景图:使用子类定义的背景资源(2x/4x) rv.setImageViewResource(R.id.widget_bg_image, getBgResourceId(bgId)); + // 携带背景ID参数,跳转编辑页时同步背景 intent.putExtra(Notes.INTENT_EXTRA_BACKGROUND_ID, bgId); + /** - * Generate the pending intent to start host for the widget + * 生成小部件点击的PendingIntent(延迟意图) + * 区分隐私模式和普通模式: + * 1. 隐私模式:显示提示文本,跳转便签列表页; + * 2. 普通模式:显示便签摘要,跳转便签编辑页。 */ PendingIntent pendingIntent = null; if (privacyMode) { + // 隐私模式:设置提示文本(“访问模式下隐藏内容”) rv.setTextViewText(R.id.widget_text, context.getString(R.string.widget_under_visit_mode)); + // 构建跳转列表页的PendingIntent pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], new Intent( context, NotesListActivity.class), PendingIntent.FLAG_UPDATE_CURRENT); } else { + // 普通模式:设置便签摘要文本 rv.setTextViewText(R.id.widget_text, snippet); + // 构建跳转编辑页的PendingIntent(FLAG_UPDATE_CURRENT:更新已有Intent的参数) pendingIntent = PendingIntent.getActivity(context, appWidgetIds[i], intent, PendingIntent.FLAG_UPDATE_CURRENT); } + // 绑定小部件文本区域的点击事件:触发上述PendingIntent rv.setOnClickPendingIntent(R.id.widget_text, pendingIntent); + // 通过小部件管理器更新当前小部件的UI appWidgetManager.updateAppWidget(appWidgetIds[i], rv); } } } + /** + * 抽象方法:获取小部件背景资源ID + * 由子类(2x/4x)实现,返回对应尺寸的背景资源ID + * @param bgId 背景色标识ID + * @return 对应尺寸的背景资源ID + */ protected abstract int getBgResourceId(int bgId); + /** + * 抽象方法:获取小部件布局ID + * 由子类(2x/4x)实现,返回对应尺寸的布局资源ID + * @return 对应尺寸的布局资源ID + */ protected abstract int getLayoutId(); + /** + * 抽象方法:获取小部件类型标识 + * 由子类(2x/4x)实现,返回对应尺寸的类型常量(Notes.TYPE_WIDGET_2X/4X) + * @return 小部件类型标识 + */ protected abstract int getWidgetType(); -} +} \ No newline at end of file