Изображение, сохраненное на SDCard, не отображается в приложении "Галерея" Android.

Я сохраняю изображение на SD-карте, и оно не появляется в приложении "Галерея", пока не сниму SD-карту и не верну ее обратно.

У вас есть идеи, почему это так?

Похоже, приложение Галерея имеет кеш, который не обновляется при сохранении файла...

На самом деле, я также хочу открыть только что сохраненное изображение в приложении "Галерея", но у меня ничего не получилось
это мой вопрос по этому вопросу.

15 ответов

Решение

Система сканирует SD-карту, когда она смонтирована, чтобы найти новые файлы изображений (и другие). Если вы программно добавляете файл, вы можете использовать этот класс:

http://developer.android.com/reference/android/media/MediaScannerConnection.html

Более простое решение - использовать статический удобный метод scanFile ():

File imageFile = ...
MediaScannerConnection.scanFile(this, new String[] { imageFile.getPath() }, new String[] { "image/jpeg" }, null);

где this вашей активности (или любого другого контекста), mime-тип необходим только в том случае, если вы используете нестандартные расширения файлов и null для необязательного обратного вызова (который нам не нужен для такого простого случая).

Мой ответ на оригинальный вопрос и всем, у кого может быть эта проблема:

У меня возникла такая же проблема, изображения в моем приложении, которые люди сохраняли на SD-карту, не сразу появлялись в их Галерее. После некоторых поисков я обнаружил, что одна строка кода вставлена ​​после кода "save to sdcard", который устранил проблему:

sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://"+ Environment.getExternalStorageDirectory())));

Обновление галереи, включая Android KITKAT

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT)
{
        Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
        File f = new File("file://"+ Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES));
        Uri contentUri = Uri.fromFile(f);
        mediaScanIntent.setData(contentUri);
        this.sendBroadcast(mediaScanIntent);
}
else
{
        sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://" + Environment.getExternalStorageDirectory())));
}

Вы также можете добавить изображение в галерею мультимедиа намеренно, посмотрите на пример кода, чтобы увидеть, как это делается:

ContentValues image = new ContentValues();

image.put(Images.Media.TITLE, imageTitle);
image.put(Images.Media.DISPLAY_NAME, imageDisplayName);
image.put(Images.Media.DESCRIPTION, imageDescription);
image.put(Images.Media.DATE_ADDED, dateTaken);
image.put(Images.Media.DATE_TAKEN, dateTaken);
image.put(Images.Media.DATE_MODIFIED, dateTaken);
image.put(Images.Media.MIME_TYPE, "image/png");
image.put(Images.Media.ORIENTATION, 0);

 File parent = imageFile.getParentFile();
 String path = parent.toString().toLowerCase();
 String name = parent.getName().toLowerCase();
 image.put(Images.ImageColumns.BUCKET_ID, path.hashCode());
 image.put(Images.ImageColumns.BUCKET_DISPLAY_NAME, name);
 image.put(Images.Media.SIZE, imageFile.length());

 image.put(Images.Media.DATA, imageFile.getAbsolutePath());

 Uri result = context.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, image);

Вот код для MediaScannerConnection:

MyMediaConnectorClient client = new MyMediaConnectorClient(newfile);
MediaScannerConnection scanner = new MediaScannerConnection(context, client);
client.setScanner(scanner);
scanner.connect();

newfile - это объект File вашего нового / сохраненного файла.

В эмуляторе есть приложение с надписью "Dev Tools"

нажмите на это и выберите "Media Scanning".. все изображения будут отсканированы

Позвольте вашему действию реализовать MediaScannerConnectionClient и добавить это к своему действию:

private void startScan() 
{ 
    if(conn!=null) conn.disconnect();  
    conn = new MediaScannerConnection(YourActivity.this,YourActivity.this); 
    conn.connect(); 
} 

@Override 
public void onMediaScannerConnected() { 
    try{
        conn.scanFile(yourImagePath, "image/*");
       } catch (java.lang.IllegalStateException e){
       }
}

@Override 
public void onScanCompleted(String path, Uri uri) { 
    conn.disconnect(); 
} 

Эта работа со мной

File file = ..... // Save file

context.sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.fromFile(file)));
 File folderGIF = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES) + "/newgif2");    //path where gif will be stored
success = folderGIF.mkdir();    //make directory
 String finalPath = folderGIF + "/test1.gif";  //path of file
.....
/* changes in gallery app if any changes in done*/
 MediaScannerConnection.scanFile(this,
                    new String[]{finalPath}, null,
                    new MediaScannerConnection.OnScanCompletedListener() {
                        public void onScanCompleted(String path, Uri uri) {
                            Log.i("ExternalStorage", "Scanned " + path + ":");
                            Log.i("ExternalStorage", "-> uri=" + uri);
                        }
                    });
sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://"+ Environment.getExternalStorageDirectory())));

Кажется, не работает на KITKAT. Выдает исключение отказа в разрешении и вылетает приложение. Итак, для этого я сделал следующее,

String path = mediaStorageDir.getPath() + File.separator
                    + "IMG_Some_name.jpg";
CameraActivity.this.sendBroadcast(new Intent(
                             Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri
                            .parse("file://" + path)));

Надеюсь, поможет.

Здесь я делюсь кодом, который может загрузить изображение в виде растрового изображения и сохранить это изображение в галерее SDCard в папке с именем приложения. Вы должны следовать этим шагам

  1. Сначала загрузите растровое изображение



     private Bitmap loadBitmap(String url) {
        try {
            InputStream in = new java.net.URL(url).openStream();
            return BitmapFactory.decodeStream(in);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
       }

  1. Также предоставьте следующее разрешение в вашем файле AndroidManifest.xml.


    uses-permission android:name="android.permission.INTERNET" 
    uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"

  1. Вот весь код, написанный в Activty, в котором мы хотим выполнить эту задачу.



     void saveMyImage(String appName, String imageUrl, String imageName) {

            Bitmap bmImg = loadBitmap(imageUrl);
            File filename;
            try {
                String path1 = android.os.Environment.getExternalStorageDirectory()
                        .toString();
                File file = new File(path1 + "/" + appName);
                if (!file.exists())
                    file.mkdirs();
                filename = new File(file.getAbsolutePath() + "/" + imageName
                        + ".jpg");
                FileOutputStream out = new FileOutputStream(filename);
                bmImg.compress(Bitmap.CompressFormat.JPEG, 90, out);
                out.flush();
                out.close();
                ContentValues image = new ContentValues();
                image.put(Images.Media.TITLE, appName);
                image.put(Images.Media.DISPLAY_NAME, imageName);
                image.put(Images.Media.DESCRIPTION, "App Image");
                image.put(Images.Media.DATE_ADDED, System.currentTimeMillis());
                image.put(Images.Media.MIME_TYPE, "image/jpg");
                image.put(Images.Media.ORIENTATION, 0);
                File parent = filename.getParentFile();
                image.put(Images.ImageColumns.BUCKET_ID, parent.toString()
                        .toLowerCase().hashCode());
                image.put(Images.ImageColumns.BUCKET_DISPLAY_NAME, parent.getName()
                        .toLowerCase());
                image.put(Images.Media.SIZE, filename.length());
                image.put(Images.Media.DATA, filename.getAbsolutePath());
                Uri result = getContentResolver().insert(
                        MediaStore.Images.Media.EXTERNAL_CONTENT_URI, image);
                Toast.makeText(getApplicationContext(),
                        "File is Saved in  " + filename, Toast.LENGTH_SHORT).show();
            } catch (Exception e) {
                e.printStackTrace();
            }

        }

  1. Надеюсь, что это может решить всю вашу проблему.

Используйте это после сохранения изображения

sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://"+ Environment.getExternalStorageDirectory())));

Мой код для MyMediaConnectorClient:

открытый класс MyMediaConnectorClient реализует MediaScannerConnectionClient {

String _fisier;
MediaScannerConnection MEDIA_SCANNER_CONNECTION;

public MyMediaConnectorClient(String nume) {
    _fisier = nume;
}

public void setScanner(MediaScannerConnection msc){
    MEDIA_SCANNER_CONNECTION = msc;
}

@Override
public void onMediaScannerConnected() {
    MEDIA_SCANNER_CONNECTION.scanFile(_fisier, null);
}

@Override
public void onScanCompleted(String path, Uri uri) {
    if(path.equals(_fisier))
        MEDIA_SCANNER_CONNECTION.disconnect();
}

}

Попробуйте этот, он будет транслировать о новом созданном изображении, поэтому ваше изображение будет видно. внутри галереи.photoFile замените фактическим путем к файлу вновь созданного изображения

private void galleryAddPicBroadCast() {
        Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
        Uri contentUri = Uri.fromFile(photoFile);
        mediaScanIntent.setData(contentUri);
        this.sendBroadcast(mediaScanIntent);
    }

Это также решит вашу проблему, если ваше изображение в галерее не отображается, вместо этого они могут отображать растровое изображение типа 404 в середине. добавьте теги, которые есть в моем коде, к вашему изображению, потому что для отображения изображения в галерее необходимы некоторые метаданные.

      String resultPath = getExternalFilesDir(Environment.DIRECTORY_PICTURES)+ 
      getString(R.string.directory) + System.currentTimeMillis() + ".jpg";

       new File(resultPath).getParentFile().mkdir();

        try {
            OutputStream fileOutputStream = new FileOutputStream(resultPath);
            savedBitmap.compress(CompressFormat.JPEG, 100, fileOutputStream);
            fileOutputStream.flush();
            fileOutputStream.close();
        } catch (IOException e2) {
            e2.printStackTrace();
        }
        savedBitmap.recycle();

        File file = new File(resultPath);
        ContentValues values = new ContentValues();
        values.put(MediaStore.Images.Media.TITLE, "Photo");
        values.put(MediaStore.Images.Media.DESCRIPTION, "Edited");
        values.put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg");
        values.put(MediaStore.Images.Media.DATE_TAKEN, System.currentTimeMillis ());
        values.put(MediaStore.Images.Media.DATE_ADDED, System.currentTimeMillis());
        values.put(MediaStore.Images.ImageColumns.BUCKET_ID, file.toString().toLowerCase(Locale.US).hashCode());
        values.put(MediaStore.Images.ImageColumns.BUCKET_DISPLAY_NAME, file.getName().toLowerCase(Locale.US));
        values.put("_data", resultPath);

        ContentResolver cr = getContentResolver();
        cr.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);




        return  resultPath;

Вам нужно дать разрешения приложению Галерея. Просто нажмите и удерживайте значок приложения галереи на главном экране и нажмите "ИНФОРМАЦИЯ О ПРИЛОЖЕНИИ", которая появляется в верхней части экрана. Это покажет настройки приложения галереи. Теперь перейдите на вкладку Permissions и включите хранилище, разрешения камеры, переключив его. Теперь перейдите в приложение родной галереи, и вы получите сохраненные изображения.

Другие вопросы по тегам