Android |内容观察者 |内容 URI 不包含资源 ID

Android | contentObserver | Content URI does not contain the resource ID

我正在尝试检测 Android 应用程序的屏幕截图。 我正在使用 contentObserver 来检测媒体目录中的更改,而不是使用 FileObserver,因为 known issue 在 Android M.

上使用它

这是代码片段:

    handlerThread = new HandlerThread("content_observer");
    handlerThread.start();
    final Handler handler = new Handler(handlerThread.getLooper()) {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
        }
    };

    getContentResolver().registerContentObserver(
            MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
            true,
            new ContentObserver(handler) {
                @Override
                public void onChange(boolean selfChange, Uri uri) {
                    Log.d(TAG, "URL : " + uri.toString());

                    //Code to query using content resolver
                    if (uri.toString().matches(MediaStore.Images.Media.EXTERNAL_CONTENT_URI.toString())) {

                        Cursor cursor = null;
                        try {
                            cursor = getContentResolver().query(uri, new String[] {
                                    MediaStore.Images.Media.DISPLAY_NAME,
                                    MediaStore.Images.Media.DATA
                            }, null, null, null);
                            if (cursor != null && cursor.moveToFirst()) {          
                             //Logic to filter if it is a screenshot file
                            }
                        } finally {
                            if (cursor != null)  {
                                cursor.close();
                            }
                        }
                    }
                    super.onChange(selfChange, uri);
                }
            }
    );

调用 onChange 时(截屏后/媒体文件夹中的任何其他更改)URI 是“content://media/external/images/media”而不是整个 URI还包含资源 ID,例如:content://media/external/images/media/12345,我尝试过的大多数设备都会出现这种情况 - Moto G3 (Android 6.0 .1), Nexus 5 (Android 6.0), 我在 Samsung S5 运行 6.0.1.

上获得了整个 URI (包括 ID)

这是一个已知问题吗?

如何获得 ID?

遍历整个媒体目录并查找是否在应用程序处于活动状态时截屏并不是一个优化的解决方案,而是在内容 URI 中获取资源 ID 可以解决问题,因为我们可以直接获取文件的路径并检测它是否是屏幕截图文件,观察者仍然会观察所有媒体内容的变化(例如,它还会观察到后台发生的 WhatsApp 图片下载)。

不幸的是,我认为您无法指望在插入 MediaStore.

中指定的表后收到正确的 Uri

这是 Android 5.1.1 的 MediaProvider#insert() 方法的来源(取自 here):

@Override
public Uri insert(Uri uri, ContentValues initialValues) {
    int match = URI_MATCHER.match(uri);

    ArrayList<Long> notifyRowIds = new ArrayList<Long>();
    Uri newUri = insertInternal(uri, match, initialValues, notifyRowIds);
    notifyMtp(notifyRowIds);

    // do not signal notification for MTP objects.
    // we will signal instead after file transfer is successful.
    if (newUri != null && match != MTP_OBJECTS) {
        getContext().getContentResolver().notifyChange(uri, null);
    }
    return newUri;
}

注意这一行:getContext().getContentResolver().notifyChange(uri, null).

上面的意思是插入到 MediaProvider 后,将发送指定插入的 Uri 的通知,它始终是 "directory" Uri。这可能是一个错误:如果他们使用 newUri 而不是 uri - 它会完全按照您的预期工作。

如果它在 Samsung 上与 Marshmallow 一起工作,那么这个问题要么在 Android 6+ 中修复,要么 Samsung 在他们的 AOSP 构建中修复它。但我不认为你真的可以指望它可靠地工作。