1
0
mirror of https://github.com/strawberrymusicplayer/strawberry synced 2024-12-31 18:47:30 +01:00
strawberry-audio-player-win.../3rdparty/taglib/fileref.cpp

470 lines
16 KiB
C++
Raw Normal View History

2018-05-10 15:29:28 +02:00
/***************************************************************************
copyright : (C) 2002 - 2008 by Scott Wheeler
email : wheeler@kde.org
copyright : (C) 2010 by Alex Novichkov
email : novichko@atnet.ru
(added APE file support)
***************************************************************************/
/***************************************************************************
* This library is free software; you can redistribute it and/or modify *
* it under the terms of the GNU Lesser General Public License version *
* 2.1 as published by the Free Software Foundation. *
* *
* This library is distributed in the hope that it will be useful, but *
* WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
* Lesser General Public License for more details. *
* *
* You should have received a copy of the GNU Lesser General Public *
* License along with this library; if not, write to the Free Software *
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA *
* 02110-1301 USA *
* *
* Alternatively, this file is available under the Mozilla Public *
* License Version 1.1. You may obtain a copy of the License at *
* http://www.mozilla.org/MPL/ *
***************************************************************************/
#include <tfile.h>
#include <tfilestream.h>
#include <tstring.h>
#include <tdebug.h>
#include <trefcounter.h>
#include "fileref.h"
#include "asffile.h"
#include "mpegfile.h"
#include "vorbisfile.h"
#include "flacfile.h"
#include "oggflacfile.h"
#include "mpcfile.h"
#include "mp4file.h"
#include "wavpackfile.h"
#include "speexfile.h"
#include "opusfile.h"
#include "trueaudiofile.h"
#include "aifffile.h"
#include "wavfile.h"
#include "apefile.h"
#include "modfile.h"
#include "s3mfile.h"
#include "itfile.h"
#include "xmfile.h"
2018-09-02 01:39:02 +02:00
#include "dsffile.h"
#include "dsdifffile.h"
2018-05-10 15:29:28 +02:00
using namespace Strawberry_TagLib::TagLib;
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
namespace {
typedef List<const FileRef::FileTypeResolver *> ResolverList;
ResolverList fileTypeResolvers;
// Detect the file type by user-defined resolvers.
File *detectByResolvers(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
2020-06-13 19:02:42 +02:00
ResolverList::ConstIterator it = fileTypeResolvers.begin();
for (; it != fileTypeResolvers.end(); ++it) {
File *file = (*it)->createFile(fileName, readAudioProperties, audioPropertiesStyle);
if (file)
return file;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
return nullptr;
2020-06-13 19:02:42 +02:00
}
// Detect the file type based on the file extension.
2018-05-10 15:29:28 +02:00
File *detectByExtension(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
2018-05-10 15:29:28 +02:00
#ifdef _WIN32
2020-06-13 19:02:42 +02:00
const String s = stream->name().toString();
2018-05-10 15:29:28 +02:00
#else
2020-06-13 19:02:42 +02:00
const String s(stream->name());
2018-05-10 15:29:28 +02:00
#endif
2020-06-13 19:02:42 +02:00
String ext;
const int pos = s.rfind(".");
if (pos != -1)
ext = s.substr(pos + 1).upper();
// If this list is updated, the method defaultFileExtensions() should also be
// updated. However at some point that list should be created at the same time
// that a default file type resolver is created.
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
if (ext.isEmpty())
2019-04-21 21:39:11 +02:00
return nullptr;
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
// .oga can be any audio in the Ogg container. So leave it to content-based detection.
if (ext == "MP3")
return new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if (ext == "OGG")
return new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "FLAC")
return new FLAC::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if (ext == "MPC")
return new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "WV")
return new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "SPX")
return new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "OPUS")
return new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "TTA")
return new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
return new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "WMA" || ext == "ASF")
return new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
return new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "WAV")
return new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "APE")
return new APE::File(stream, readAudioProperties, audioPropertiesStyle);
// module, nst and wow are possible but uncommon extensions
if (ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
return new Mod::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "S3M")
return new S3M::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "IT")
return new IT::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "XM")
return new XM::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "DFF" || ext == "DSDIFF")
return new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "DSF")
return new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
return nullptr;
2020-06-13 19:02:42 +02:00
}
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
// Detect the file type based on the actual content of the stream.
File *detectByContent(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
File *file = nullptr;
2020-06-13 19:02:42 +02:00
if (MPEG::File::isSupported(stream))
file = new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
else if (Ogg::Vorbis::File::isSupported(stream))
file = new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
else if (Ogg::FLAC::File::isSupported(stream))
file = new Ogg::FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
else if (FLAC::File::isSupported(stream))
file = new FLAC::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
else if (MPC::File::isSupported(stream))
file = new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
else if (WavPack::File::isSupported(stream))
file = new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
else if (Ogg::Speex::File::isSupported(stream))
file = new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
else if (Ogg::Opus::File::isSupported(stream))
file = new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
else if (TrueAudio::File::isSupported(stream))
file = new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
else if (MP4::File::isSupported(stream))
file = new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
else if (ASF::File::isSupported(stream))
file = new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
else if (RIFF::AIFF::File::isSupported(stream))
file = new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
else if (RIFF::WAV::File::isSupported(stream))
file = new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
else if (APE::File::isSupported(stream))
file = new APE::File(stream, readAudioProperties, audioPropertiesStyle);
else if (DSDIFF::File::isSupported(stream))
file = new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
else if (DSF::File::isSupported(stream))
file = new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
// isSupported() only does a quick check, so double check the file here.
if (file) {
if (file->isValid())
return file;
else
delete file;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
return nullptr;
2020-06-13 19:02:42 +02:00
}
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
// Internal function that supports FileRef::create().
// This looks redundant, but necessary in order not to change the previous
// behavior of FileRef::create().
File *createInternal(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
2020-06-13 19:02:42 +02:00
File *file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle);
if (file) return file;
2018-05-10 15:29:28 +02:00
#ifdef _WIN32
2020-06-13 19:02:42 +02:00
const String s = fileName.toString();
2018-05-10 15:29:28 +02:00
#else
2020-06-13 19:02:42 +02:00
const String s(fileName);
2018-05-10 15:29:28 +02:00
#endif
2020-06-13 19:02:42 +02:00
String ext;
const int pos = s.rfind(".");
if (pos != -1)
ext = s.substr(pos + 1).upper();
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
if (ext.isEmpty())
2019-04-21 21:39:11 +02:00
return nullptr;
2020-06-13 19:02:42 +02:00
if (ext == "MP3")
return new MPEG::File(fileName, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if (ext == "OGG")
return new Ogg::Vorbis::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "OGA") {
/* .oga can be any audio in the Ogg container. First try FLAC, then Vorbis. */
File *file_flac = new Ogg::FLAC::File(fileName, readAudioProperties, audioPropertiesStyle);
if (file_flac->isValid())
return file_flac;
delete file_flac;
return new Ogg::Vorbis::File(fileName, readAudioProperties, audioPropertiesStyle);
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
if (ext == "FLAC")
return new FLAC::File(fileName, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if (ext == "MPC")
return new MPC::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "WV")
return new WavPack::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "SPX")
return new Ogg::Speex::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "OPUS")
return new Ogg::Opus::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "TTA")
return new TrueAudio::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
return new MP4::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "WMA" || ext == "ASF")
return new ASF::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
return new RIFF::AIFF::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "WAV")
return new RIFF::WAV::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "APE")
return new APE::File(fileName, readAudioProperties, audioPropertiesStyle);
// module, nst and wow are possible but uncommon extensions
if (ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
return new Mod::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "S3M")
return new S3M::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "IT")
return new IT::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "XM")
return new XM::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "DFF" || ext == "DSDIFF")
return new DSDIFF::File(fileName, readAudioProperties, audioPropertiesStyle);
if (ext == "DSF")
return new DSF::File(fileName, readAudioProperties, audioPropertiesStyle);
return nullptr;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
} // namespace
2018-05-10 15:29:28 +02:00
2020-06-13 19:02:42 +02:00
class FileRef::FileRefPrivate : public RefCounter {
public:
2020-06-15 00:11:52 +02:00
FileRefPrivate() : file(nullptr), stream(nullptr) {}
2018-05-10 15:29:28 +02:00
2020-06-15 21:55:05 +02:00
~FileRefPrivate() override {
2018-05-10 15:29:28 +02:00
delete file;
delete stream;
}
2020-06-13 19:02:42 +02:00
File *file;
2018-05-10 15:29:28 +02:00
IOStream *stream;
};
////////////////////////////////////////////////////////////////////////////////
// public members
////////////////////////////////////////////////////////////////////////////////
FileRef::FileRef() : d(new FileRefPrivate()) {}
2018-05-10 15:29:28 +02:00
FileRef::FileRef(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) : d(new FileRefPrivate()) {
2018-05-10 15:29:28 +02:00
parse(fileName, readAudioProperties, audioPropertiesStyle);
}
2020-06-13 19:02:42 +02:00
FileRef::FileRef(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) : d(new FileRefPrivate()) {
2018-05-10 15:29:28 +02:00
parse(stream, readAudioProperties, audioPropertiesStyle);
}
2020-06-13 19:02:42 +02:00
FileRef::FileRef(File *file) : d(new FileRefPrivate()) {
2018-05-10 15:29:28 +02:00
d->file = file;
}
2020-06-13 19:02:42 +02:00
FileRef::FileRef(const FileRef &ref) : d(ref.d) {
2018-05-10 15:29:28 +02:00
d->ref();
}
2020-06-13 19:02:42 +02:00
FileRef::~FileRef() {
2020-06-13 19:02:42 +02:00
if (d->deref())
2018-05-10 15:29:28 +02:00
delete d;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
Tag *FileRef::tag() const {
2020-06-13 19:02:42 +02:00
if (isNull()) {
2018-05-10 15:29:28 +02:00
debug("FileRef::tag() - Called without a valid file.");
2019-04-21 21:39:11 +02:00
return nullptr;
2018-05-10 15:29:28 +02:00
}
return d->file->tag();
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
AudioProperties *FileRef::audioProperties() const {
2020-06-13 19:02:42 +02:00
if (isNull()) {
2018-05-10 15:29:28 +02:00
debug("FileRef::audioProperties() - Called without a valid file.");
2019-04-21 21:39:11 +02:00
return nullptr;
2018-05-10 15:29:28 +02:00
}
return d->file->audioProperties();
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
File *FileRef::file() const {
2018-05-10 15:29:28 +02:00
return d->file;
}
2020-06-13 19:02:42 +02:00
bool FileRef::save() {
2020-06-13 19:02:42 +02:00
if (isNull()) {
2018-05-10 15:29:28 +02:00
debug("FileRef::save() - Called without a valid file.");
return false;
}
return d->file->save();
2018-05-10 15:29:28 +02:00
}
const FileRef::FileTypeResolver *FileRef::addFileTypeResolver(const FileRef::FileTypeResolver *resolver) { // static
2018-05-10 15:29:28 +02:00
fileTypeResolvers.prepend(resolver);
return resolver;
}
2020-06-13 19:02:42 +02:00
StringList FileRef::defaultFileExtensions() {
2018-05-10 15:29:28 +02:00
StringList l;
l.append("ogg");
l.append("flac");
l.append("oga");
l.append("mp3");
l.append("mpc");
l.append("wv");
l.append("spx");
l.append("tta");
l.append("m4a");
l.append("m4r");
l.append("m4b");
l.append("m4p");
l.append("3g2");
l.append("mp4");
l.append("m4v");
l.append("wma");
l.append("asf");
l.append("aif");
l.append("aiff");
l.append("wav");
l.append("ape");
l.append("mod");
2020-06-13 19:02:42 +02:00
l.append("module"); // alias for "mod"
l.append("nst"); // alias for "mod"
l.append("wow"); // alias for "mod"
2018-05-10 15:29:28 +02:00
l.append("s3m");
l.append("it");
l.append("xm");
2018-09-02 01:39:02 +02:00
l.append("dsf");
l.append("dff");
2020-06-13 19:02:42 +02:00
l.append("dsdiff"); // alias for "dff"
2018-05-10 15:29:28 +02:00
return l;
2018-05-10 15:29:28 +02:00
}
2020-06-13 19:02:42 +02:00
bool FileRef::isNull() const {
2018-05-10 15:29:28 +02:00
return (!d->file || !d->file->isValid());
}
2020-06-13 19:02:42 +02:00
FileRef &FileRef::operator=(const FileRef &ref) {
2018-05-10 15:29:28 +02:00
FileRef(ref).swap(*this);
return *this;
}
2020-06-13 19:02:42 +02:00
void FileRef::swap(FileRef &ref) {
2018-05-10 15:29:28 +02:00
using std::swap;
swap(d, ref.d);
}
2020-06-13 19:02:42 +02:00
bool FileRef::operator==(const FileRef &ref) const {
2018-05-10 15:29:28 +02:00
return (ref.d->file == d->file);
}
2020-06-13 19:02:42 +02:00
bool FileRef::operator!=(const FileRef &ref) const {
2018-05-10 15:29:28 +02:00
return (ref.d->file != d->file);
}
File *FileRef::create(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) { // static
2018-05-10 15:29:28 +02:00
return createInternal(fileName, readAudioProperties, audioPropertiesStyle);
}
////////////////////////////////////////////////////////////////////////////////
// private members
////////////////////////////////////////////////////////////////////////////////
void FileRef::parse(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
2018-05-10 15:29:28 +02:00
// Try user-defined resolvers.
d->file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle);
2020-06-13 19:02:42 +02:00
if (d->file)
2018-05-10 15:29:28 +02:00
return;
// Try to resolve file types based on the file extension.
d->stream = new FileStream(fileName);
d->file = detectByExtension(d->stream, readAudioProperties, audioPropertiesStyle);
2020-06-13 19:02:42 +02:00
if (d->file)
2018-05-10 15:29:28 +02:00
return;
// At last, try to resolve file types based on the actual content.
d->file = detectByContent(d->stream, readAudioProperties, audioPropertiesStyle);
2020-06-13 19:02:42 +02:00
if (d->file)
2018-05-10 15:29:28 +02:00
return;
// Stream have to be closed here if failed to resolve file types.
delete d->stream;
d->stream = nullptr;
2018-05-10 15:29:28 +02:00
}
void FileRef::parse(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
2018-05-10 15:29:28 +02:00
// User-defined resolvers won't work with a stream.
// Try to resolve file types based on the file extension.
d->file = detectByExtension(stream, readAudioProperties, audioPropertiesStyle);
2020-06-13 19:02:42 +02:00
if (d->file)
2018-05-10 15:29:28 +02:00
return;
// At last, try to resolve file types based on the actual content of the file.
d->file = detectByContent(stream, readAudioProperties, audioPropertiesStyle);
2018-05-10 15:29:28 +02:00
}