Android: чи можна відображати ескізи відео?


91

Я створив програму для відеозйомки з діалоговим вікном бібліотеки. У діалоговому вікні бібліотеки відображається список записаних відео, де кожен елемент складається з піктограми, заголовка відео, тегів та інформації про місце розташування наступним чином:

текст заміщення

Хтось знає, чи можна замінити піктограми на ескізи відео (попередній перегляд одного кадру)?

Дякую!


Будь-яка відповідь на це [ stackoverflow.com/questions/16190374/…
Зробіть це простим

Відповіді:


71

Якщо ви використовуєте API 2.0 або новішу версію, це буде працювати.

int id = **"The Video's ID"**
ImageView iv = (ImageView ) convertView.findViewById(R.id.imagePreview);
ContentResolver crThumb = getContentResolver();
BitmapFactory.Options options=new BitmapFactory.Options();
options.inSampleSize = 1;
Bitmap curThumb = MediaStore.Video.Thumbnails.getThumbnail(crThumb, id, MediaStore.Video.Thumbnails.MICRO_KIND, options);
iv.setImageBitmap(curThumb);

9
То що саме id?
phunehehe

1
Ви можете запитати MediaStore для відео на телефоні. "Ідентифікатор" - це лише одна з частин інформації, яку ви запитуєте. Докладніше про MediaStore див. На веб-сайті developer.android.com/reference/android/provider/…
Грег Ціммерс

4
Здивований, здається, всі працюють. Я спробував це, але curThumb закінчується нулем.
BlueVoodoo

7
як щодо відео з URL-адреси?
jayellos

будь ласка, дайте відповідь на це [ stackoverflow.com/questions/16190374/…
Зробіть це простим

91

якщо ви не можете або не можете пройти курсор, і якщо у вас є лише шляхи або об'єкти File, ви можете використовувати, оскільки API рівня 8 (2.2) відкритий статичний Bitmap createVideoThumbnail (String filePath, int kind)

Документація для Android

Наступний код працює ідеально:

Bitmap bMap = ThumbnailUtils.createVideoThumbnail(file.getAbsolutePath(), MediaStore.Video.Thumbnails.MICRO_KIND);

6
при спробі створити мініатюру я отримую нуль. Я тник мій шлях, можливо, не в порядку? myPath = "/ external / video / media /
14180

Це працює як магія. Навіть коли я t have my video ID. For better quality use надягаю MediaStore.Video.Thumbnails.FULL_SCREEN_KIND`
Самі Елтамаві

Дивно, що це теж не працює ;-( відео існує в БД, я можу отримати ім'я / розмір, але не мініатюру
Томас Деко

haythem souussi, оскільки це не шлях, це Урі, вам потрібно перетворити його на шлях.
Надір Новрузов

Це працює, але повертає зображення з неправильної частини відео. Я хочу найперший кадр, але через 5-6 секунд? Будь-які ідеї?
speedynomads

38

Використання класу:

import android.provider.MediaStore.Video.Thumbnails;

Ми можемо отримати два розміри мініатюр попереднього перегляду з відео:

Thumbnails.MICRO_KIND для 96 х 96

Thumbnails.MINI_KIND для 512 x 384 пікселів

Це приклад коду:

String filePath = "/sdcard/DCIM/Camera/my_video.mp4"; //change the location of your file!

ImageView imageview_mini = (ImageView)findViewById(R.id.thumbnail_mini);
ImageView imageview_micro = (ImageView)findViewById(R.id.thumbnail_micro);

Bitmap bmThumbnail;

//MICRO_KIND, size: 96 x 96 thumbnail
bmThumbnail = ThumbnailUtils.createVideoThumbnail(filePath, Thumbnails.MICRO_KIND);
imageview_micro.setImageBitmap(bmThumbnail);
     
// MINI_KIND, size: 512 x 384 thumbnail 
bmThumbnail = ThumbnailUtils.createVideoThumbnail(filePath, Thumbnails.MINI_KIND);
imageview_mini.setImageBitmap(bmThumbnail);

Якщо у мене є таке посилання для шляху до файлу, це не спрацювало б, тому що я намагаюся встановити для нього перегляд зображення, і він нічого не показує ... це шлях до файлу "http: / /unknown.com/v3- 1aox9d1 .mp4 ", очевидно, є реальним доменом, але чи цей шлях не підтримує ескізи?
Lion789

Щоб використовувати клас ThumbnailUtils, потрібно зберегти файл на диск, використовуючи метод: ThumbnailUtils.createVideoThumbnail ()
Jorgesys

Дякую, як мені тоді отримати новий шлях до файлу для створення великого пальця?
Lion789

Чому доза не працює на Android 4 і вище?
Criss

1
Привіт, Кріс, у мене є 3 пристрої 4.1 4.2.2 і 5.0, і це працює без проблем, опублікуй запитання про свою проблему та якийсь код, і я міг би тобі допомогти.
Jorgesys

19

В даний час я використовую такий код:

Bitmap bMap = ThumbnailUtils.createVideoThumbnail(file.getAbsolutePath(), MediaStore.Video.Thumbnails.MICRO_KIND);

Але я знайшов краще рішення з бібліотекою Glide з наступним кодом (він також кешує ваше зображення та має кращу продуктивність, ніж попередній підхід)

Glide.with(context)
                .load(uri)
                .placeholder(R.drawable.ic_video_place_holder)
                .into(imageView);

19

Я справді пропоную вам скористатися Glide бібліотекою . Це один з найефективніших способів створення та відображення ескізу відео для локального відеофайлу.

Просто додайте цей рядок у файл gradle:

compile 'com.github.bumptech.glide:glide:3.7.0'

І це стане настільки просто, як:

String filePath = "/storage/emulated/0/Pictures/example_video.mp4";

Glide  
    .with( context )
    .load( Uri.fromFile( new File( filePath ) ) )
    .into( imageViewGifAsBitmap );

Ви можете знайти більше інформації тут: https://futurestud.io/blog/glide-displaying-gifs-and-videos

Ура!


4
Glide працює лише з місцевими відео, а не з Url-відео, хіба це не простий спосіб
Lutaaya Huzaifah Idris

6

Це рішення буде працювати для будь-якої версії Android. Доведено, що він працює в 1.5 та 2.2. Це не інше рішення "Це для Android 2.0+". Я знайшов це через сторінку колекції дошки повідомлень електронної пошти і не можу знайти оригінальне посилання. Вся заслуга належить оригінальному плакату.

У вашому додатку ви можете використовувати це, зателефонувавши:

Bitmap bm = getVideoFrame(VideoStringUri);

Десь у його власній функції (поза OnCreate, тощо) вам знадобиться:

private Bitmap getVideoFrame(String uri) {
        MediaMetadataRetriever retriever = new MediaMetadataRetriever();
        try {
            retriever.setMode(MediaMetadataRetriever.MODE_CAPTURE_FRAME_ONLY);
            retriever.setDataSource(uri);
            return retriever.captureFrame();
        } catch (IllegalArgumentException ex) {
            ex.printStackTrace();
        } catch (RuntimeException ex) {
            ex.printStackTrace();
        } finally {
            try {
                retriever.release();
            } catch (RuntimeException ex) {
            }
        }
        return null;
    }

У вашій папці src вам потрібен новий підкаталог android / media, в якому буде розміщений клас (скопійований із самого джерела android), що дозволяє використовувати цю функцію. Цю частину не слід міняти, перейменовувати чи розміщувати де-небудь ще. MediaMetadataRetriever.java повинна знаходитися в android.media у вихідній папці, щоб все це працювало.

/*
 * Copyright (C) 2008 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package android.media;

import java.io.FileDescriptor;
import java.io.FileNotFoundException;
import java.io.IOException;

import android.content.ContentResolver;
import android.content.Context;
import android.content.res.AssetFileDescriptor;
import android.graphics.Bitmap;
import android.net.Uri;

/**
 * MediaMetadataRetriever class provides a unified interface for retrieving
 * frame and meta data from an input media file. {@hide}
 */
public class MediaMetadataRetriever {
    static {
        System.loadLibrary("media_jni");
        native_init();
    }

    // The field below is accessed by native methods
    private int mNativeContext;

    public MediaMetadataRetriever() {
        native_setup();
    }

    /**
     * Call this method before setDataSource() so that the mode becomes
     * effective for subsequent operations. This method can be called only once
     * at the beginning if the intended mode of operation for a
     * MediaMetadataRetriever object remains the same for its whole lifetime,
     * and thus it is unnecessary to call this method each time setDataSource()
     * is called. If this is not never called (which is allowed), by default the
     * intended mode of operation is to both capture frame and retrieve meta
     * data (i.e., MODE_GET_METADATA_ONLY | MODE_CAPTURE_FRAME_ONLY). Often,
     * this may not be what one wants, since doing this has negative performance
     * impact on execution time of a call to setDataSource(), since both types
     * of operations may be time consuming.
     * 
     * @param mode
     *            The intended mode of operation. Can be any combination of
     *            MODE_GET_METADATA_ONLY and MODE_CAPTURE_FRAME_ONLY: 1.
     *            MODE_GET_METADATA_ONLY & MODE_CAPTURE_FRAME_ONLY: For neither
     *            frame capture nor meta data retrieval 2.
     *            MODE_GET_METADATA_ONLY: For meta data retrieval only 3.
     *            MODE_CAPTURE_FRAME_ONLY: For frame capture only 4.
     *            MODE_GET_METADATA_ONLY | MODE_CAPTURE_FRAME_ONLY: For both
     *            frame capture and meta data retrieval
     */
    public native void setMode(int mode);

    /**
     * @return the current mode of operation. A negative return value indicates
     *         some runtime error has occurred.
     */
    public native int getMode();

    /**
     * Sets the data source (file pathname) to use. Call this method before the
     * rest of the methods in this class. This method may be time-consuming.
     * 
     * @param path
     *            The path of the input media file.
     * @throws IllegalArgumentException
     *             If the path is invalid.
     */
    public native void setDataSource(String path)
            throws IllegalArgumentException;

    /**
     * Sets the data source (FileDescriptor) to use. It is the caller's
     * responsibility to close the file descriptor. It is safe to do so as soon
     * as this call returns. Call this method before the rest of the methods in
     * this class. This method may be time-consuming.
     * 
     * @param fd
     *            the FileDescriptor for the file you want to play
     * @param offset
     *            the offset into the file where the data to be played starts,
     *            in bytes. It must be non-negative
     * @param length
     *            the length in bytes of the data to be played. It must be
     *            non-negative.
     * @throws IllegalArgumentException
     *             if the arguments are invalid
     */
    public native void setDataSource(FileDescriptor fd, long offset, long length)
            throws IllegalArgumentException;

    /**
     * Sets the data source (FileDescriptor) to use. It is the caller's
     * responsibility to close the file descriptor. It is safe to do so as soon
     * as this call returns. Call this method before the rest of the methods in
     * this class. This method may be time-consuming.
     * 
     * @param fd
     *            the FileDescriptor for the file you want to play
     * @throws IllegalArgumentException
     *             if the FileDescriptor is invalid
     */
    public void setDataSource(FileDescriptor fd)
            throws IllegalArgumentException {
        // intentionally less than LONG_MAX
        setDataSource(fd, 0, 0x7ffffffffffffffL);
    }

    /**
     * Sets the data source as a content Uri. Call this method before the rest
     * of the methods in this class. This method may be time-consuming.
     * 
     * @param context
     *            the Context to use when resolving the Uri
     * @param uri
     *            the Content URI of the data you want to play
     * @throws IllegalArgumentException
     *             if the Uri is invalid
     * @throws SecurityException
     *             if the Uri cannot be used due to lack of permission.
     */
    public void setDataSource(Context context, Uri uri)
            throws IllegalArgumentException, SecurityException {
        if (uri == null) {
            throw new IllegalArgumentException();
        }

        String scheme = uri.getScheme();
        if (scheme == null || scheme.equals("file")) {
            setDataSource(uri.getPath());
            return;
        }

        AssetFileDescriptor fd = null;
        try {
            ContentResolver resolver = context.getContentResolver();
            try {
                fd = resolver.openAssetFileDescriptor(uri, "r");
            } catch (FileNotFoundException e) {
                throw new IllegalArgumentException();
            }
            if (fd == null) {
                throw new IllegalArgumentException();
            }
            FileDescriptor descriptor = fd.getFileDescriptor();
            if (!descriptor.valid()) {
                throw new IllegalArgumentException();
            }
            // Note: using getDeclaredLength so that our behavior is the same
            // as previous versions when the content provider is returning
            // a full file.
            if (fd.getDeclaredLength() < 0) {
                setDataSource(descriptor);
            } else {
                setDataSource(descriptor, fd.getStartOffset(),
                        fd.getDeclaredLength());
            }
            return;
        } catch (SecurityException ex) {
        } finally {
            try {
                if (fd != null) {
                    fd.close();
                }
            } catch (IOException ioEx) {
            }
        }
        setDataSource(uri.toString());
    }

    /**
     * Call this method after setDataSource(). This method retrieves the meta
     * data value associated with the keyCode.
     * 
     * The keyCode currently supported is listed below as METADATA_XXX
     * constants. With any other value, it returns a null pointer.
     * 
     * @param keyCode
     *            One of the constants listed below at the end of the class.
     * @return The meta data value associate with the given keyCode on success;
     *         null on failure.
     */
    public native String extractMetadata(int keyCode);

    /**
     * Call this method after setDataSource(). This method finds a
     * representative frame if successful and returns it as a bitmap. This is
     * useful for generating a thumbnail for an input media source.
     * 
     * @return A Bitmap containing a representative video frame, which can be
     *         null, if such a frame cannot be retrieved.
     */
    public native Bitmap captureFrame();

    /**
     * Call this method after setDataSource(). This method finds the optional
     * graphic or album art associated (embedded or external url linked) the
     * related data source.
     * 
     * @return null if no such graphic is found.
     */
    public native byte[] extractAlbumArt();

    /**
     * Call it when one is done with the object. This method releases the memory
     * allocated internally.
     */
    public native void release();

    private native void native_setup();

    private static native void native_init();

    private native final void native_finalize();

    @Override
    protected void finalize() throws Throwable {
        try {
            native_finalize();
        } finally {
            super.finalize();
        }
    }

    public static final int MODE_GET_METADATA_ONLY = 0x01;
    public static final int MODE_CAPTURE_FRAME_ONLY = 0x02;

    /*
     * Do not change these values without updating their counterparts in
     * include/media/mediametadataretriever.h!
     */
    public static final int METADATA_KEY_CD_TRACK_NUMBER = 0;
    public static final int METADATA_KEY_ALBUM = 1;
    public static final int METADATA_KEY_ARTIST = 2;
    public static final int METADATA_KEY_AUTHOR = 3;
    public static final int METADATA_KEY_COMPOSER = 4;
    public static final int METADATA_KEY_DATE = 5;
    public static final int METADATA_KEY_GENRE = 6;
    public static final int METADATA_KEY_TITLE = 7;
    public static final int METADATA_KEY_YEAR = 8;
    public static final int METADATA_KEY_DURATION = 9;
    public static final int METADATA_KEY_NUM_TRACKS = 10;
    public static final int METADATA_KEY_IS_DRM_CRIPPLED = 11;
    public static final int METADATA_KEY_CODEC = 12;
    public static final int METADATA_KEY_RATING = 13;
    public static final int METADATA_KEY_COMMENT = 14;
    public static final int METADATA_KEY_COPYRIGHT = 15;
    public static final int METADATA_KEY_BIT_RATE = 16;
    public static final int METADATA_KEY_FRAME_RATE = 17;
    public static final int METADATA_KEY_VIDEO_FORMAT = 18;
    public static final int METADATA_KEY_VIDEO_HEIGHT = 19;
    public static final int METADATA_KEY_VIDEO_WIDTH = 20;
    public static final int METADATA_KEY_WRITER = 21;
    public static final int METADATA_KEY_MIMETYPE = 22;
    public static final int METADATA_KEY_DISCNUMBER = 23;
    public static final int METADATA_KEY_ALBUMARTIST = 24;
    // Add more here...
}

У мене це не працює .. помилка в System.loadLibrary ("media_jni");
DArkO

1
Я можу підтвердити, що це не працює. Мені була потрібна і ця можливість. Це не буде працювати, оскільки він використовує власні системні дзвінки, які звичайна програма не має дозволів на використання.
Енді,

MediaMetadataRetrieverпідтримується з рівня API 10
Asahi

MediaMetadataRetriever - це другий блок коду. Існує можливість дозволити API до 10 (який був недоступний на момент написання статті) отримувати доступ до коду з програми замість системи. Можливі власні системні дзвінки, але для їх використання вам знадобиться грубе розуміння системи. Здається, багато в чому проблема полягає в неправильній реалізації наданого джерела.
Покинутий візок

@LoungeKatt чи можна дозволити йому знімати кілька зображень одного і того ж відео кілька разів?
розробник android

6

Спробуйте це, це працює для мене

RequestOptions requestOptions = new RequestOptions();
 Glide.with(getContext())
      .load("video_url")
      .apply(requestOptions)
      .thumbnail(Glide.with(getContext()).load("video_url"))
      .into("yourimageview");

5

Android 1.5 та 1.6 не пропонують цих мініатюр, але 2.0 пропонує, як видно з офіційних приміток до випуску :

ЗМІ

  • Тепер MediaScanner генерує ескізи для всіх зображень, коли вони вставляються в MediaStore.
  • Новий мініатюрний API для отримання ескізів зображень та відео на вимогу.

1

Я відповідаю на це питання пізно, але сподіваюся, це допоможе іншому кандидатові, який стикається з такою ж проблемою.

Я використав два методи для завантаження ескізу для списку відео, перший був

    Bitmap bmThumbnail;
    bmThumbnail = ThumbnailUtils.createVideoThumbnail(FILE_PATH
                    + videoList.get(position),
            MediaStore.Video.Thumbnails.MINI_KIND);

    if (bmThumbnail != null) {
        Log.d("VideoAdapter","video thumbnail found");
        holder.imgVideo.setImageBitmap(bmThumbnail);
    } else {
        Log.d("VideoAdapter","video thumbnail not found");
    }

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

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

 Glide
            .with( mContext )
            .load( Uri.fromFile( new File( FILE_PATH+videoList.get(position) ) ) )
            .into( holder.imgVideo );

Я порекомендував пізніше рішення для показу ескізу зі списком відео. Дякую


-4

Це код для ескізу відео в прямому ефірі.

public class LoadVideoThumbnail extends AsyncTask<Object, Object, Bitmap>{

        @Override
        protected Bitmap doInBackground(Object... params) {try {

            String mMediaPath = "http://commonsware.com/misc/test2.3gp";
            Log.e("TEST Chirag","<< thumbnail doInBackground"+ mMediaPath);
            FileOutputStream out;
            File land=new File(Environment.getExternalStorageDirectory().getAbsoluteFile()
                            +"/portland.jpg");

                Bitmap bitmap = ThumbnailUtils.createVideoThumbnail(mMediaPath, MediaStore.Video.Thumbnails.MICRO_KIND);
                        ByteArrayOutputStream stream = new ByteArrayOutputStream();
                        bitmap.compress(Bitmap.CompressFormat.JPEG, 100, stream);
                        byte[] byteArray = stream.toByteArray();

                        out=new  FileOutputStream(land.getPath());
                        out.write(byteArray);
                        out.close();
                 return bitmap;

            } catch (FileNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        return null;
            }
        @Override
        protected void onPostExecute(Bitmap result) {
            // TODO Auto-generated method stub
            super.onPostExecute(result);
            if(result != null){
                 ((ImageView)findViewById(R.id.imageView1)).setImageBitmap(result);
            }
            Log.e("TEST Chirag","====> End");
        }

    }

2
я отримую нуль за Bitmap bitmap = ThumbnailUtils.createVideoThumbnail(mMediaPath, MediaStore.Video.Thumbnails.MICRO_KIND);Примітка, що всі параметри встановлені
Чуло

1
нульове значення в Bitmap bitmap = ThumbnailUtils.createVideoThumbnail (mMediaPath, MediaStore.Video.Thumbnails.MICRO_KIND);
Прасад
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.