android - зберегти зображення в галерею


94

у мене є програма з галереєю зображень, і я хочу, щоб користувач міг зберегти її у власній галереї. Я створив меню опцій одним голосом "зберегти", щоб це дозволило, але проблема в тому ... як я можу зберегти зображення в галерею?

це мій код:

@Override
        public boolean onOptionsItemSelected(MenuItem item) {
            // Handle item selection
            switch (item.getItemId()) {
            case R.id.menuFinale:

                imgView.setDrawingCacheEnabled(true);
                Bitmap bitmap = imgView.getDrawingCache();
                File root = Environment.getExternalStorageDirectory();
                File file = new File(root.getAbsolutePath()+"/DCIM/Camera/img.jpg");
                try 
                {
                    file.createNewFile();
                    FileOutputStream ostream = new FileOutputStream(file);
                    bitmap.compress(CompressFormat.JPEG, 100, ostream);
                    ostream.close();
                } 
                catch (Exception e) 
                {
                    e.printStackTrace();
                }



                return true;
            default:
                return super.onOptionsItemSelected(item);
            }
        }

я не впевнений у цій частині коду:

File root = Environment.getExternalStorageDirectory();
                File file = new File(root.getAbsolutePath()+"/DCIM/Camera/img.jpg");

чи правильно зберігати в галереї? на жаль код не працює :(


Ви вирішили це питання? Ви можете поділитися зі мною
user3233280

я також маю ту ж проблему stackoverflow.com/questions/21951558 / ...
user3233280

Для тих з вас, у кого все ще виникають проблеми із збереженням файлу, можливо, це пов’язано з тим, що ваша URL-адреса містить заборонені символи, такі як «?», «:» Та «-» Видаліть їх, і це має спрацювати. Це поширена помилка в закордонних пристроях та емуляторах Android. Детальніше про це дивіться тут: stackoverflow.com/questions/11394616/…
ChallengeПрийнято

Прийнята відповідь трохи застаріла у 2019 році. Я написав тут оновлену відповідь: stackoverflow.com/questions/36624756/…
Бао Лей,

Відповіді:


171
MediaStore.Images.Media.insertImage(getContentResolver(), yourBitmap, yourTitle , yourDescription);

Колишній код додасть зображення в кінці галереї. Якщо ви хочете змінити дату так, щоб вона відображалася на початку, або будь-які інші метадані, дивіться код нижче (Cortesy з SK, самкіртон ):

https://gist.github.com/samkirton/0242ba81d7ca00b475b9

/**
 * Android internals have been modified to store images in the media folder with 
 * the correct date meta data
 * @author samuelkirton
 */
public class CapturePhotoUtils {

    /**
     * A copy of the Android internals  insertImage method, this method populates the 
     * meta data with DATE_ADDED and DATE_TAKEN. This fixes a common problem where media 
     * that is inserted manually gets saved at the end of the gallery (because date is not populated).
     * @see android.provider.MediaStore.Images.Media#insertImage(ContentResolver, Bitmap, String, String)
     */
    public static final String insertImage(ContentResolver cr, 
            Bitmap source, 
            String title, 
            String description) {

        ContentValues values = new ContentValues();
        values.put(Images.Media.TITLE, title);
        values.put(Images.Media.DISPLAY_NAME, title);
        values.put(Images.Media.DESCRIPTION, description);
        values.put(Images.Media.MIME_TYPE, "image/jpeg");
        // Add the date meta data to ensure the image is added at the front of the gallery
        values.put(Images.Media.DATE_ADDED, System.currentTimeMillis());
        values.put(Images.Media.DATE_TAKEN, System.currentTimeMillis());

        Uri url = null;
        String stringUrl = null;    /* value to be returned */

        try {
            url = cr.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);

            if (source != null) {
                OutputStream imageOut = cr.openOutputStream(url);
                try {
                    source.compress(Bitmap.CompressFormat.JPEG, 50, imageOut);
                } finally {
                    imageOut.close();
                }

                long id = ContentUris.parseId(url);
                // Wait until MINI_KIND thumbnail is generated.
                Bitmap miniThumb = Images.Thumbnails.getThumbnail(cr, id, Images.Thumbnails.MINI_KIND, null);
                // This is for backward compatibility.
                storeThumbnail(cr, miniThumb, id, 50F, 50F,Images.Thumbnails.MICRO_KIND);
            } else {
                cr.delete(url, null, null);
                url = null;
            }
        } catch (Exception e) {
            if (url != null) {
                cr.delete(url, null, null);
                url = null;
            }
        }

        if (url != null) {
            stringUrl = url.toString();
        }

        return stringUrl;
    }

    /**
     * A copy of the Android internals StoreThumbnail method, it used with the insertImage to
     * populate the android.provider.MediaStore.Images.Media#insertImage with all the correct
     * meta data. The StoreThumbnail method is private so it must be duplicated here.
     * @see android.provider.MediaStore.Images.Media (StoreThumbnail private method)
     */
    private static final Bitmap storeThumbnail(
            ContentResolver cr,
            Bitmap source,
            long id,
            float width, 
            float height,
            int kind) {

        // create the matrix to scale it
        Matrix matrix = new Matrix();

        float scaleX = width / source.getWidth();
        float scaleY = height / source.getHeight();

        matrix.setScale(scaleX, scaleY);

        Bitmap thumb = Bitmap.createBitmap(source, 0, 0,
            source.getWidth(),
            source.getHeight(), matrix,
            true
        );

        ContentValues values = new ContentValues(4);
        values.put(Images.Thumbnails.KIND,kind);
        values.put(Images.Thumbnails.IMAGE_ID,(int)id);
        values.put(Images.Thumbnails.HEIGHT,thumb.getHeight());
        values.put(Images.Thumbnails.WIDTH,thumb.getWidth());

        Uri url = cr.insert(Images.Thumbnails.EXTERNAL_CONTENT_URI, values);

        try {
            OutputStream thumbOut = cr.openOutputStream(url);
            thumb.compress(Bitmap.CompressFormat.JPEG, 100, thumbOut);
            thumbOut.close();
            return thumb;
        } catch (FileNotFoundException ex) {
            return null;
        } catch (IOException ex) {
            return null;
        }
    }
}

23
Це зберігає зображення, але до кінця галереї, хоча, коли ви робите знімок камерою, воно зберігається зверху. Як я можу зберегти зображення у верхній частині галереї?
eric.itzhak

19
Зверніть увагу, що ви також повинні додати <koristi-дозвіл android: name = "android.permission.WRITE_EXTERNAL_STORAGE" /> до вашого manifext.xml.
Kyle Clegg

3
Зображення не зберігаються у верхній частині галереї, оскільки внутрішньо insertImage не додає жодних метаданих дати. Будь ласка, ознайомтесь з цим GIST: gist.github.com/0242ba81d7ca00b475b9.git - це точна копія методу insertImage, але він додає дату мета дати, щоб переконатися, що зображення додано в передній частині галереї.
S-K

6
Ось правильне посилання на GIST, згадане вище (необхідне для видалення .gitв кінці)
minipif

2
Як змінити назву папки файлу?
Копі Брайант

48

Насправді, ви можете зберегти свою картинку в будь-якому місці. Якщо ви хочете економити у публічному просторі, щоб мати доступ будь-яка інша програма, використовуйте цей код:

storageDir = new File(
    Environment.getExternalStoragePublicDirectory(
        Environment.DIRECTORY_PICTURES
    ), 
    getAlbumName()
);

Зображення не потрапляє в альбом. Для цього вам потрібно викликати скан:

private void galleryAddPic() {
    Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
    File f = new File(mCurrentPhotoPath);
    Uri contentUri = Uri.fromFile(f);
    mediaScanIntent.setData(contentUri);
    this.sendBroadcast(mediaScanIntent);
}

Ви можете знайти більше інформації на https://developer.android.com/training/camera/photobasics.html#TaskGallery


1
Це приємне просте рішення, оскільки нам не потрібно змінювати всю реалізацію, і ми можемо створити власну папку для програм.
Hugo Gresse

2
Надсилання трансляції може бути марною тратою ресурсів, коли ви можете сканувати лише файл: stackoverflow.com/a/5814533/43051 .
Джеремі Рейно

2
Куди ви насправді передаєте растрове зображення?
Даніель Рейханян,

Environment.getExternalStoragePublicDirectoryі Intent.ACTION_MEDIA_SCANNER_SCAN_FILEзастаріли зараз ...
Михайло Абизов

22

Я спробував багато речей, щоб дозволити цій роботі працювати над Зефіром та льодяником. Нарешті, я закінчив переміщенням збереженого зображення в папку DCIM (нова програма сканування зображень програми Google Photo, лише якщо вони, мабуть, знаходяться всередині цієї папки)

public static File createImageFile() throws IOException {
    // Create an image file name
    String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss")
         .format(System.currentTimeInMillis());
    File storageDir = new File(Environment
         .getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM) + "/Camera/");
    if (!storageDir.exists())
        storageDir.mkdirs();
    File image = File.createTempFile(
            timeStamp,                   /* prefix */
            ".jpeg",                     /* suffix */
            storageDir                   /* directory */
    );
    return image;
}

А потім стандартний код для сканування файлів, який ви також можете знайти на веб-сайті Google Developers .

public static void addPicToGallery(Context context, String photoPath) {
    Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
    File f = new File(photoPath);
    Uri contentUri = Uri.fromFile(f);
    mediaScanIntent.setData(contentUri);
    context.sendBroadcast(mediaScanIntent);
}

Пам’ятайте, що ця папка не могла бути присутнім на всіх пристроях у світі, і що починаючи з Marshmallow (API 23), вам потрібно запитати у користувача дозвіл на WRITE_EXTERNAL_STORAGE.


1
Дякуємо за інформацію щодо Google Фото.
Джеремі Рейно

1
Це єдине рішення, яке добре пояснює. Ніхто інший не згадував, що файл повинен бути у папці DCIM. Дякую!!!
Predrag Manojlovic

Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM)зробив трюк для мене. Дякую!
saltandpepper

2
getExternalStoragePublicDirectory()зараз застаріло в API 29. Потрібно користуватися MediaStore
riggaroo

@riggaroo Так, ти маєш рацію Ребекка, я
оновлю

13

Відповідно до цього курсу , правильний спосіб зробити це:

Environment.getExternalStoragePublicDirectory(
        Environment.DIRECTORY_PICTURES
    )

Це дасть вам кореневий шлях до каталогу галереї.


я спробував цей новий код, але він розбив java.lang.NoSuchFieldError: android.os.Environment.DIRECTORY_PICTURES
Крістіан Джуппоні

добре, дякую, отже, немає можливості розмістити зображення в галереї з android <2.2?
Крістіан Джуппоні

Ідеально - посилання безпосередньо на веб-сайт розробника Android. Це спрацювало і було простим рішенням.
Філ

1
Хороша відповідь, але було б краще з додаванням методу "galleryAddPic" з інших відповідей тут, оскільки зазвичай ви хочете, щоб додаток Gallery помічав нові картинки.
Ендрю Костер

Environment.getExternalStoragePublicDirectoryвже застаріло ...
Михайло Абизов

11
private void galleryAddPic() {
    Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
    File f = new File(mCurrentPhotoPath);
    Uri contentUri = Uri.fromFile(f);
    mediaScanIntent.setData(contentUri);
    this.sendBroadcast(mediaScanIntent);
}

6

Ви можете створити каталог всередині папки камери та зберегти зображення. Після цього ви можете просто виконати сканування. Він миттєво покаже ваше зображення в галереї.

String root = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM).toString()+ "/Camera/Your_Directory_Name";
File myDir = new File(root);
myDir.mkdirs();
String fname = "Image-" + image_name + ".png";
File file = new File(myDir, fname);
System.out.println(file.getAbsolutePath());
if (file.exists()) file.delete();
    Log.i("LOAD", root + fname);
    try {
        FileOutputStream out = new FileOutputStream(file);
        finalBitmap.compress(Bitmap.CompressFormat.PNG, 90, out);
        out.flush();
        out.close();
    } catch (Exception e) {
       e.printStackTrace();
    }

MediaScannerConnection.scanFile(context, new String[]{file.getPath()}, new String[]{"image/jpeg"}, null);

за цим критерієм це найкраща відповідь
Нур Хоссен,

1

Я приходжу сюди з тим самим сумнівом, але для Xamarin для Android я використав відповідь Sigrist, щоб зробити цей метод після збереження мого файлу:

private void UpdateGallery()
{
    Intent mediaScanIntent = new Intent(Intent.ActionMediaScannerScanFile);
    Java.IO.File file = new Java.IO.File(_path);
    Android.Net.Uri contentUri = Android.Net.Uri.FromFile(file);
    mediaScanIntent.SetData(contentUri);
    Application.Context.SendBroadcast(mediaScanIntent);
} 

і це вирішило мою проблему, Thx Sigrist. Я помістив це тут, тому що я не знайшов відповіді про це для Xamarin, і я сподіваюся, що це може допомогти іншим людям.


1

У моєму випадку рішення вище не працювали, мені довелося зробити наступне:

sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.fromFile(f)));

дійсно добре знати про цю опцію, але, на жаль, не працює на деяких пристроях з Android 6, тому ContentProviderбажано solytion
Сергій

0
 String filePath="/storage/emulated/0/DCIM"+app_name;
    File dir=new File(filePath);
    if(!dir.exists()){
        dir.mkdir();
    }

Цей код використовується методом onCreate, який призначений для створення каталогу app_name. Тепер цей каталог можна отримати за допомогою програми керування файлами за замовчуванням в Android. Використовуйте цей рядок filePath скрізь, де потрібно для встановлення цільової папки. Я впевнений, що цей метод працює і на Android 7, тому що я тестував його, отже, він може працювати і на інших версіях Android.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.