Fixing some issues with old id3v1 genre numbers in some rare mp3s.

Added a generic way of getting all of taglibs supported files.
This commit is contained in:
Daniel Önnerby 2008-04-06 22:10:36 +00:00
parent ccd8ef9e5c
commit d4fd4122cc
2 changed files with 286 additions and 141 deletions

View File

@ -36,19 +36,25 @@
#include "TagReaderTaglib.h"
#include <tlist.h>
#include <fileref.h>
#include <tfile.h>
#include <tag.h>
#include <mpegfile.h>
#include <id3v2tag.h>
#include <id3v2frame.h>
#include <id3v2header.h>
#include <id3v1tag.h>
#include <id3v1genres.h>
#include <audioproperties.h>
#include <attachedpictureframe.h>
#include <commentsframe.h>
#include <toolkit/tlist.h>
#include <toolkit/tfile.h>
#include <taglib/tag.h>
#include <taglib/fileref.h>
#include <taglib/audioproperties.h>
#include <mpeg/mpegfile.h>
#include <mpeg/id3v1/id3v1tag.h>
#include <mpeg/id3v1/id3v1genres.h>
#include <mpeg/id3v2/id3v2tag.h>
#include <mpeg/id3v2/id3v2header.h>
#include <mpeg/id3v2/id3v2frame.h>
#include <mpeg/id3v2/frames/attachedpictureframe.h>
#include <mpeg/id3v2/frames/commentsframe.h>
#include <taglib/ogg/oggfile.h>
#include <vector>
#include <boost/format.hpp>
@ -65,15 +71,22 @@ TagReaderTaglib::~TagReaderTaglib(void){
}
void TagReaderTaglib::Destroy() {
// do nothing! we use a static reference for every call to GetMetaDataReader()
delete this;
}
bool TagReaderTaglib::CanReadTag(const wchar_t *extension){
bool TagReaderTaglib::CanReadTag(const utfchar *extension){
if(extension){
// std::wstring ext(extension);
if( std::wstring(_T("mp3"))==extension ) {
utfstring ext(extension);
boost::algorithm::to_lower(ext); // Convert to lower case
if( ext==UTF("mp3") ||
ext==UTF("ogg") ||
ext==UTF("flac") ||
ext==UTF("ape") ||
ext==UTF("mpc")
) {
return true;
}
}
@ -82,16 +95,85 @@ bool TagReaderTaglib::CanReadTag(const wchar_t *extension){
}
bool TagReaderTaglib::ReadTag(musik::core::Track *track){
TagLib::ID3v2::FrameFactory::instance()->setDefaultTextEncoding(TagLib::String::UTF16);
if(std::wstring(_T("mp3"))==track->GetValue("extension")){
const utfchar *extension = track->GetValue("extension");
if(extension){
utfstring ext(extension);
boost::algorithm::to_lower(ext); // Convert to lower case
if(ext==UTF("mp3"))
if(this->GetID3v2Tag(track))
return true;
if(ext==UTF("ogg"))
if(this->GetOGGTag(track))
return true;
}
if(this->GetGenericTag(track)) // secondary: try generic way
return true;
return false;
}
bool TagReaderTaglib::GetOGGTag(musik::core::Track *track){
/*
TagLib::Ogg::File file(track->GetValue("path"));
*/
return false;
}
bool TagReaderTaglib::GetGenericTag(musik::core::Track *track){
TagLib::FileRef oFile(track->GetValue("path"));
TagLib::Tag *tag = oFile.tag();
TagLib::AudioProperties *oAudioProperties = oFile.audioProperties();
if(tag){
// TITLE
if(!tag->title().isEmpty()){
this->SetTagValue("title",tag->title(),track);
}else{
this->SetTagValue("title",track->GetValue("filename"),track);
}
// ALBUM
this->SetTagValue("album",tag->album(),track);
// ARTISTS
this->SetSlashSeparatedValues("artist",tag->album(),track);
// GENRES
this->SetTagValue("genre",tag->genre(),track);
// COMMENT
this->SetTagValue("comment",tag->comment(),track);
// TRACK
this->SetTagValue("track",tag->track(),track);
// TRACK
this->SetTagValue("year",tag->year(),track);
this->SetAudioProperties(oAudioProperties,track);
return true;
}
return false;
}
bool TagReaderTaglib::GetID3v2Tag(musik::core::Track *track){
#ifdef UTF_WIDECHAR
TagLib::ID3v2::FrameFactory::instance()->setDefaultTextEncoding(TagLib::String::UTF16);
#else
TagLib::ID3v2::FrameFactory::instance()->setDefaultTextEncoding(TagLib::String::UTF8);
#endif
TagLib::MPEG::File oFile(track->GetValue("path"));
TagLib::ID3v2::Tag *oTagv2 = oFile.ID3v2Tag();
TagLib::AudioProperties *oAudioProperties = oFile.audioProperties();
if(oTagv2){
TagLib::AudioProperties *oAudioProperties = oFile.audioProperties();
// Successfully found ID3V2-tag
//////////////////////////////////////////////////////////////////////////////
@ -137,17 +219,36 @@ bool TagReaderTaglib::ReadTag(musik::core::Track *track){
if( !genreString.isEmpty() ){
// There is no need to worry about duplicates, since the Track will not save duplicates.
// TODO: This should also handle old ID3v1 genre numbers
int numberLength(0);
bool isNumber(true);
for(TagLib::String::ConstIterator charIt=genreString.begin();isNumber && charIt!=genreString.end();++charIt){
isNumber = *charIt >= '0' && *charIt <= '9';
++numberLength;
}
if(!isNumber && numberLength>0){
// Genre starting with a number
if(genreString.substr(numberLength,1)==" "){
// If the genre number is appended by a space, then this is a id3v1 genre number
int genreNumber = genreString.substr(0,numberLength).toInt();
if(genreNumber>= 0 && genreNumber<= 255)
this->SetTagValue("genre",TagLib::ID3v1::genre(genreNumber),track);
// Remove the prepending genre number
genreString = genreString.substr(numberLength+1);
}
}
if(isNumber){
// If the genre is only a number, look for id3v1 genres
int genreNumber = genreString.toInt();
if(genreNumber>= 0 && genreNumber<= 255)
genreString = TagLib::ID3v1::genre(genreNumber);
}
if( !genreString.isEmpty() ){
this->SetTagValue("genre",genreString,track);
}
}
}
@ -162,10 +263,7 @@ bool TagReaderTaglib::ReadTag(musik::core::Track *track){
//////////////////////////////////////////////////////////////////////////////
// Audio properties
if(oAudioProperties){
std::wstring duration = boost::str(boost::wformat(_T("%1%"))%oAudioProperties->length());
this->SetTagValue("duration",duration,track);
}
this->SetAudioProperties(oAudioProperties,track);
//////////////////////////////////////////////////////////////////////////////
// Comments
@ -205,10 +303,10 @@ bool TagReaderTaglib::ReadTag(musik::core::Track *track){
return true;
}
}
/* }
// Try the "generic" way
/* TagLib::FileRef oFile(oMedia->sFileFullPath.c_str());
TagLib::FileRef oFile(oMedia->sFileFullPath.c_str());
TagLib::Tag *oTag = oFile.tag();
TagLib::AudioProperties *oAudioProperties = oFile.audioProperties();
if(oTag && oAudioProperties){
@ -232,6 +330,11 @@ void TagReaderTaglib::SetTagValue(const char* key,const wchar_t* string,musik::c
track->SetValue(key,string);
}
void TagReaderTaglib::SetTagValue(const char* key,const int tagInt,musik::core::Track *track){
std::wstring temp = boost::str(boost::wformat(_T("%1%"))%tagInt);
track->SetValue(key,temp.c_str());
}
void TagReaderTaglib::SetTagValues(const char* key,const TagLib::ID3v2::FrameList &frame,musik::core::Track *track){
using namespace musik::core;
@ -263,11 +366,7 @@ bool TagReaderTaglib::getStandardTags(musik::core::Track &track,TagLib::Tag *oTa
return true;
}
*/
void TagReaderTaglib::SetSlashSeparatedValues(const char* key,const TagLib::ID3v2::FrameList &frame,musik::core::Track *track){
using namespace musik::core;
if(!frame.isEmpty()){
for(TagLib::ID3v2::FrameList::ConstIterator frameValue=frame.begin();frameValue!=frame.end();++frameValue){
TagLib::String tagString = (*frameValue)->toString();
void TagReaderTaglib::SetSlashSeparatedValues(const char* key,TagLib::String &tagString,musik::core::Track *track){
if( !tagString.isEmpty() ){
std::wstring value(tagString.begin(),tagString.end());
std::vector<std::wstring> splitValues;
@ -279,7 +378,43 @@ void TagReaderTaglib::SetSlashSeparatedValues(const char* key,const TagLib::ID3v
}
}
}
void TagReaderTaglib::SetSlashSeparatedValues(const char* key,const TagLib::ID3v2::FrameList &frame,musik::core::Track *track){
using namespace musik::core;
if(!frame.isEmpty()){
for(TagLib::ID3v2::FrameList::ConstIterator frameValue=frame.begin();frameValue!=frame.end();++frameValue){
TagLib::String tagString = (*frameValue)->toString();
/* if( !tagString.isEmpty() ){
std::wstring value(tagString.begin(),tagString.end());
std::vector<std::wstring> splitValues;
boost::algorithm::split(splitValues,value,boost::algorithm::is_any_of(_T("/")));
for(std::vector<std::wstring>::iterator theValue=splitValues.begin();theValue!=splitValues.end();++theValue){
track->SetValue(key,theValue->c_str());
}
}*/
this->SetSlashSeparatedValues(key,tagString,track);
}
}
}
void TagReaderTaglib::SetAudioProperties(TagLib::AudioProperties *audioProperties,musik::core::Track *track){
if(audioProperties){
std::wstring duration = boost::str(boost::wformat(_T("%1%"))%audioProperties->length());
this->SetTagValue("duration",duration,track);
int bitrate( audioProperties->bitrate() );
if(bitrate){
std::wstring temp( boost::str(boost::wformat(_T("%1%"))%bitrate ) );
this->SetTagValue("bitrate",temp,track);
}
int channels( audioProperties->channels() );
if(channels){
std::wstring temp( boost::str(boost::wformat(_T("%1%"))%channels ) );
this->SetTagValue("channels",temp,track);
}
}
}

View File

@ -38,12 +38,15 @@
#include "stdafx.h"
#include <tlist.h>
#include <fileref.h>
#include <tfile.h>
#include <tag.h>
#include <mpegfile.h>
#include <id3v2tag.h>
#include <toolkit/tlist.h>
#include <toolkit/tfile.h>
#include <taglib/tag.h>
#include <taglib/fileref.h>
#include <taglib/audioproperties.h>
#include <mpeg/id3v2/id3v2tag.h>
#include <set>
@ -54,15 +57,22 @@ class TagReaderTaglib : public musik::core::Plugin::IMetaDataReader {
TagReaderTaglib(void);
virtual ~TagReaderTaglib(void);
bool ReadTag(musik::core::Track *track);
virtual bool CanReadTag(const wchar_t *extension);
virtual bool CanReadTag(const utfchar *extension);
virtual void Destroy();
private:
// bool getStandardTags(musik::core::Track &track,TagLib::Tag *oTag,TagLib::AudioProperties *oAudioProperties);
void SetTagValue(const char* key,const wchar_t* string,musik::core::Track *track);
void SetTagValue(const char* key,const TagLib::String tagString,musik::core::Track *track);
void SetTagValue(const char* key,const int tagInt,musik::core::Track *track);
void SetTagValues(const char* key,const TagLib::ID3v2::FrameList &frame,musik::core::Track *track);
void SetAudioProperties(TagLib::AudioProperties *audioProperties,musik::core::Track *track);
void SetSlashSeparatedValues(const char* key,const TagLib::ID3v2::FrameList &frame,musik::core::Track *track);
void SetSlashSeparatedValues(const char* key,TagLib::String &tagString,musik::core::Track *track);
bool GetID3v2Tag(musik::core::Track *track);
bool GetGenericTag(musik::core::Track *track);
bool GetOGGTag(musik::core::Track *track);
};