Android составляет список музыки по папкам и воспроизводит их

Я разрабатываю музыкальный плеер для Android, который уже есть на рынке. Пользователи просят добавить представление «Папка», чтобы перечислить все папки с музыкой на их смартфоне, и я хочу его развить.

MediaStore уже знает, по каким путям следует следовать (потому что ему нужно знать их, чтобы каждый раз повторно сканировать SD-карту), поэтому мне было интересно, есть ли способ получить эти пути и использовать их для показа музыки пользователю.

Другой вариант (но мне он не нравится) для получения этих путей — запросить в MediaStore всю музыку в библиотеке и зациклиться на них, чтобы получить общие базовые пути. Но если у пользователя более 1/2k музыки, это будет огромный и неоптимизированный путь.

Вы знаете, как я могу это решить?

Я также думал о перечислении всех папок и файлов на SD-карте, но столкнулся с некоторой рекурсивной проблемой, потому что не хочу показывать:

  • не аудиофайл (уже сделано с помощью FileFilter)
  • не показывать пустые папки (уже сделано, но есть рекурсивная проблема)
  • не показывать папки с файлом .nomedia (это говорит мне, что медиафайлов нет, но есть рекурсивная проблема). Также многие папки, созданные другими приложениями, не содержат файл .nomedia, поэтому я должен их показать

Что Вы думаете об этом?

ОБНОВЛЕНИЕ: я хочу сделать что-то вроде этого: https://play.google.com/store/apps/details?id=cyberniko.musicFolderPlayer Я хочу показать папки, в которых есть музыка, и разрешить пользователю начать воспроизведение музыки в выбранных папках.


person StErMi    schedule 05.06.2012    source источник


Ответы (2)


Как я уже сказал, одно из возможных решений — начать сканирование всех каталогов с SDCard с помощью этого FileFilter, который я написал:

Этот AudioFilter применяется к каталогу, который возвращает все файлы, являющиеся песней (проверка расширений), и все каталоги, содержащие песню, но не содержащие .nomedia.

Это решение в любом случае нужно пропустить, потому что загрузка списка каталогов занимает слишком много времени (из-за рекурсивного алгоритма)

package com.designfuture.music.util;

import java.io.File;
import java.io.FileFilter;
import java.io.FilenameFilter;
import java.util.ArrayList;

import com.designfuture.framework.util.LogHelper;

public class AudioFileFilter implements FileFilter {

    protected static final String TAG = "AudioFileFilter";
    /**
     * allows Directories
     */
    private final boolean allowDirectories;

    public AudioFileFilter( boolean allowDirectories) {
        this.allowDirectories = allowDirectories;
    }

    public AudioFileFilter() {
        this(true);
    }

    @Override
    public boolean accept(File f) {
        if ( f.isHidden() || !f.canRead() ) {
            return false;
        }

        if ( f.isDirectory() ) {
            return checkDirectory( f );
        }
        return checkFileExtension( f );
    }

    private boolean checkFileExtension( File f ) {
        String ext = getFileExtension(f);
        if ( ext == null) return false;
        try {
            if ( SupportedFileFormat.valueOf(ext.toUpperCase()) != null ) {
                return true;
            }
        } catch(IllegalArgumentException e) {
            //Not known enum value
            return false;    
        }
        return false; 
    }

    private boolean checkDirectory( File dir ) {
        if ( !allowDirectories ) {
            return false;
        } else {
            final ArrayList<File> subDirs = new ArrayList<File>();
            int songNumb = dir.listFiles( new FileFilter() {

                @Override
                public boolean accept(File file) {
                    if ( file.isFile() ) {
                        if ( file.getName().equals( ".nomedia" ) )
                            return false;

                        return checkFileExtension( file );
                    } else if ( file.isDirectory() ){
                        subDirs.add( file );
                        return false;
                    } else
                        return false;
                }
            } ).length;

            if ( songNumb > 0 ) {
                LogHelper.i(TAG, "checkDirectory: dir " + dir.toString() + " return true con songNumb -> " + songNumb );
                return true;
            }

            for( File subDir: subDirs ) {
                if ( checkDirectory( subDir ) ) {
                    LogHelper.i(TAG, "checkDirectory [for]: subDir " + subDir.toString() + " return true" );
                    return true;
                }
            }
            return false;
        }       
    }

    private boolean checkFileExtension( String fileName ) {
        String ext = getFileExtension(fileName);
        if ( ext == null) return false;
        try {
            if ( SupportedFileFormat.valueOf(ext.toUpperCase()) != null ) {
                return true;
            }
        } catch(IllegalArgumentException e) {
            //Not known enum value
            return false;    
        }
        return false; 
    }

    public String getFileExtension( File f ) {
        return getFileExtension( f.getName() );
    }

    public String getFileExtension( String fileName ) {
        int i = fileName.lastIndexOf('.');
        if (i > 0) {
            return fileName.substring(i+1);
        } else 
            return null;
    }

    /**
     * Files formats currently supported by Library
     */
    public enum SupportedFileFormat
    {
        _3GP("3gp"),
        MP4("mp4"),
        M4A("m4a"),
        AAC("aac"),
        TS("ts"),
        FLAC("flac"),
        MP3("mp3"),
        MID("mid"),
        XMF("xmf"),
        MXMF("mxmf"),
        RTTTL("rtttl"),
        RTX("rtx"),
        OTA("ota"),
        IMY("imy"),
        OGG("ogg"),
        MKV("mkv"),
        WAV("wav");

        private String filesuffix;

        SupportedFileFormat( String filesuffix ) {
            this.filesuffix = filesuffix;
        }

        public String getFilesuffix() {
            return filesuffix;
        }
    }

}
person StErMi    schedule 05.06.2012

MediaStore уже знает, по каким путям следует следовать (потому что ему нужно знать их, чтобы каждый раз повторно сканировать SD-карту), поэтому мне было интересно, есть ли способ получить эти пути и использовать их для показа музыки пользователю.

MediaStore сканирует все внешнее хранилище, насколько я знаю, а не только некоторые определенные корневые папки.

person CommonsWare    schedule 06.06.2012