当前位置: 首页 > 知识库问答 >
问题:

如何使用光标将封面艺术嵌入到音频文件中?

叶文博
2023-03-14

我正在开发一个非常小的音频播放器。问题是:在许多问题之后,我设法用从外部存储器的任何文件夹中检索到的歌曲构建了一个列表视图,并将它们与标题、艺术家和专辑名一起列出。现在我想在ListView中添加封面相册。封面必须取自歌曲音频文件中嵌入的图像。

我尝试使用MediaMetadataRetriever,但无法获取每个文件的完整Uri,因此无法为其设置数据源。我怎么才能拿到封面?如果我有字节数组,我会使用位图工厂。。。但我没有。

顺便说一句,这是我的代码...在活动中,这是在外部存储中搜索音频文件并将其放入列表的空格:

public void retrieveAudioFiles(){
        songsList = new SongsList();

        Uri sd = Audio.Media.EXTERNAL_CONTENT_URI;
        String[] cols = {Audio.Media.TITLE,Audio.Media.ARTIST,Audio.Media.ALBUM};
        String where = Audio.Media.IS_MUSIC;
        Cursor audioCursor = getContentResolver().query(sd,cols,where,null,null);

        while (audioCursor.moveToNext()){
            int posColTitle = audioCursor.getColumnIndex(Audio.Media.TITLE);
            int posColArtist = audioCursor.getColumnIndex(Audio.Media.ARTIST);
            int posColAlbum = audioCursor.getColumnIndex(Audio.Media.ALBUM);

            String songTitle = audioCursor.getString(posColTitle);
            String songArtist = audioCursor.getString(posColArtist);
            String songAlbum = audioCursor.getString(posColAlbum);

            songsList.add(new Song(songTitle,songArtist,songAlbum));
            }
        audioCursor.close();
    }

这是适配器:

public class SongsAdapter extends BaseAdapter {

    private SongsList songsList;
    private LayoutInflater songInf;

    public SongsAdapter(Context c, SongsList theSongs){
        super();  
        songsList=theSongs;
        songInf=LayoutInflater.from(c);
    }

    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return songsList.size();
    }

    @Override
    public Object getItem(int position) {
        // TODO Auto-generated method stub
        return songsList.get(position);
    }

    @Override
    public long getItemId(int position) {
        // TODO Auto-generated method stub
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        // TODO Auto-generated method stub
        RowWrapper wrapper;
        if (convertView == null)
        {
            convertView = songInf.inflate(
                R.layout.song_row, null);
            wrapper = new RowWrapper(convertView);
            convertView.setTag(wrapper);
        }
        else
        {
            wrapper = (RowWrapper) convertView.getTag();
        }
        Song song = (Song) getItem(position);
        wrapper.populate(song);

        return convertView;
    }

    private static class RowWrapper
    {
        private TextView titleTextView;
        private TextView artistTextView;
        private TextView albumTextView;
        //private ImageView coverImageView;

        public RowWrapper(View convertView)
        {
            titleTextView = (TextView) convertView.findViewById(R.id.textTitle);
            artistTextView = (TextView) convertView.findViewById(R.id.textArtist);
            albumTextView = (TextView) convertView.findViewById(R.id.textAlbum);
            //coverImageView = (ImageView) convertView.findViewById(R.id.smallCover);
        }

        public void populate(Song song)
        {
            titleTextView.setText(song.title);
            artistTextView.setText(song.artist);
            albumTextView.setText(song.album);
            //if (song.cover != null)
            //coverImageView.setImageBitmap(song.cover);
        }
    }

}

这是歌曲课:

public class Song {

    public String title="";
    public String artist="";
    public String album="";
    public Bitmap cover=null;

    public Song(String t, String ar, String al){
        title=t;
        artist=ar;
        album=al;
        //cover=c;
    }

    public Song(){

    }

}

这是Song sList类(一个简单的ArrayList的歌曲):

public class SongsList extends ArrayList<Song> {

    public SongsList(){
        super();
    }

}

这是舱单:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.audioplayer"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="14"
        android:targetSdkVersion="19" />
    <uses-permission android:name="android.permission.WAKE_LOCK"/>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.audioplayer.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

这是ListActivity的单行布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/LinearLayout1"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_weight="1"
    android:orientation="vertical"
    android:onClick="songPicked" >

    <ImageView
        android:id="@+id/smallCover"
        android:layout_width="60dp"
        android:layout_height="60dp"
        android:contentDescription="@string/coverDescription"
        android:src="@drawable/no_cover" />

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content" >

        <TextView
            android:id="@+id/labelTitle"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@string/labelTitle" />

        <TextView
            android:id="@+id/textTitle"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:paddingLeft="5dp"
            android:text="@string/textTitle" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content" >

        <TextView
            android:id="@+id/labelArtist"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginLeft="0dp"
            android:text="@string/labelArtist" />

        <TextView
            android:id="@+id/textArtist"
            android:layout_width="150dp"
            android:layout_height="wrap_content"
            android:paddingLeft="5dp"
            android:text="@string/textArtist" />

        <TextView
            android:id="@+id/labelAlbum"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginLeft="10dp"
            android:text="@string/labelAlbum" />

        <TextView
            android:id="@+id/textAlbum"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:paddingLeft="5dp"
            android:text="@string/textAlbum" />
    </LinearLayout>

</LinearLayout>

希望你能帮我。。。

共有1个答案

秦永望
2023-03-14

我自己做的...

我在Song类中为id添加了一个长参数,为path添加了一个Uri参数。

活动中,就在songsL之前ist.add(新歌(......))(我在构造函数中添加了Bitmap参数),我添加了以下说明:

int posColId = audioCursor.getColumnIndex(Audio.Media._ID);
long songId = audioCursor.getLong(posColId);
Uri songUri = ContentUris.withAppendedId(Audio.Media.EXTERNAL_CONTENT_URI,songId);
String[] dataColumn = {Audio.Media.DATA};
Cursor coverCursor = getContentResolver().query(songUri, dataColumn, null, null, null);
coverCursor.moveToFirst();
int dataIndex = coverCursor.getColumnIndex(Audio.Media.DATA);
String filePath = coverCursor.getString(dataIndex);
coverCursor.close();
MediaMetadataRetriever retriever = new MediaMetadataRetriever();
retriever.setDataSource(filePath);
byte[] coverBytes = retriever.getEmbeddedPicture();
Bitmap songCover;
if (coverBytes!=null) //se l'array di byte non è vuoto, crea una bitmap
    songCover = BitmapFactory.decodeByteArray(coverBytes, 0, coverBytes.length);
else
    songCover=null;

然后我取消了Song类和适配器中的指令注释。

 类似资料:
  • 本文向大家介绍HTML5中如何嵌入音频?相关面试题,主要包含被问及HTML5中如何嵌入音频?时的应答技巧和注意事项,需要的朋友参考一下 HTML5支持MP3、Wav和Ogg格式的音频,下面是在网页中嵌入音频的简单实例:  

  • 本文向大家介绍如何在HTML5页面中嵌入音频和视频?相关面试题,主要包含被问及如何在HTML5页面中嵌入音频和视频?时的应答技巧和注意事项,需要的朋友参考一下 嵌入视频 嵌入音频

  • 但是我将ffmpeg更新到了最新版本(ffmpeg version git-2012-06-16-809d71d),现在在这个版本中参数不起作用。 请告诉我如何使用将新音频添加到视频(不是混合)中。

  • 问题内容: 我最终想要做的是从用户的麦克风中进行录音,并在完成后将文件上传到服务器。到目前为止,我已经设法使用以下代码将数据流传递到元素: 如何从那开始,到录制到文件? 问题答案: 它允许您在浏览器中录制音频,然后为您提供导出和下载录制内容的选项。 您可以查看该页面的源代码以找到指向javascript的链接,但总而言之,有一个对象包含一个方法和一个方法。

  • 我只从使用的youtube视频中提取音频。我想在下载后将元数据(即艺人名称和歌名)写入mp3文件。我尝试从以下代码开始: 这段代码的输出是: 如您所见,代码将元数据添加到文件名中,而不是添加到文件中。将其写入文件是无用的,因为该文件在进程完成后会被删除。我希望将此元数据写入文件,以便在查看文件夹中的歌曲时,它将如下所示: 这种格式对我很有用,因为我可以直接输入这些文件到iTunes,元数据将是完整

  • 问题内容: 我正在尝试为我的Swift应用创建视频播放器,但始终收到“无法解析的标识符AVPlayerViewController”的错误。我想念什么? 我是一个初学者,我可能不得不问外行几千次。我可能一直在互联网上搜寻某天的视频,以了解如何将Youtube视频嵌入到我的应用中,但没有结果。如果您可以将我引向一个很棒的教程! 谢谢! 问题答案: Xcode 8.2•Swift 3.0.2 Xcod