You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdemultimedia/tdefile-plugins/ogg/tdefile_ogg.cpp

358 lines
11 KiB

/* This file is part of the KDE project
* Copyright (C) 2001, 2002 Rolf Magnus <ramagnus@kde.org>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public
* License as published by the Free Software Foundation version 2.
*
* This program 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
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; see the file COPYING. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*
* $Id$
*/
#include "tdefile_ogg.h"
#include "vcedit.h"
#include <tqcstring.h>
#include <tqfile.h>
#include <tqdatetime.h>
#include <tqdict.h>
#include <tqvalidator.h>
#include <tqfileinfo.h>
#include <kdebug.h>
#include <kurl.h>
#include <kprocess.h>
#include <tdelocale.h>
#include <kgenericfactory.h>
#include <ksavefile.h>
#include <ogg/ogg.h>
#include <vorbis/codec.h>
#include <vorbis/vorbisfile.h>
#include <sys/stat.h>
#include <unistd.h>
// known translations for common ogg/vorbis keys
// from http://www.ogg.org/ogg/vorbis/doc/v-comment.html
static const char* const knownTranslations[] = {
I18N_NOOP("Title"),
I18N_NOOP("Version"),
I18N_NOOP("Album"),
I18N_NOOP("Tracknumber"),
I18N_NOOP("Artist"),
I18N_NOOP("Organization"),
I18N_NOOP("Description"),
I18N_NOOP("Genre"),
I18N_NOOP("Date"),
I18N_NOOP("Location"),
I18N_NOOP("Copyright")
// I18N_NOOP("Isrc") // dunno what an Isrc number is, the link is broken
};
K_EXPORT_COMPONENT_FACTORY(tdefile_ogg, KGenericFactory<KOggPlugin>("tdefile_ogg"))
KOggPlugin::KOggPlugin( TQObject *parent, const char *name,
const TQStringList &args )
: KFilePlugin( parent, name, args )
{
kdDebug(7034) << "ogg plugin\n";
KFileMimeTypeInfo* info = addMimeTypeInfo( "audio/vorbis" );
KFileMimeTypeInfo::GroupInfo* group = 0;
// comment group
group = addGroupInfo(info, "Comment", i18n("Comment"));
setAttributes(group, KFileMimeTypeInfo::Addable |
KFileMimeTypeInfo::Removable);
KFileMimeTypeInfo::ItemInfo* item = 0;
item = addItemInfo(group, "Artist", i18n("Artist"), TQVariant::String);
setHint(item, KFileMimeTypeInfo::Author);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Title", i18n("Title"), TQVariant::String);
setHint(item, KFileMimeTypeInfo::Name);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Album", i18n("Album"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Genre", i18n("Genre"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Tracknumber", i18n("Track Number"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Date", i18n("Date"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Description", i18n("Description"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Organization", i18n("Organization"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Location", i18n("Location"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
item = addItemInfo(group, "Copyright", i18n("Copyright"), TQVariant::String);
setAttributes(item, KFileMimeTypeInfo::Modifiable);
addVariableInfo(group, TQVariant::String, KFileMimeTypeInfo::Addable |
KFileMimeTypeInfo::Removable |
KFileMimeTypeInfo::Modifiable);
// technical group
group = addGroupInfo(info, "Technical", i18n("Technical Details"));
setAttributes(group, 0);
addItemInfo(group, "Version", i18n("Version"), TQVariant::Int);
addItemInfo(group, "Channels", i18n("Channels"), TQVariant::Int);
item = addItemInfo(group, "Sample Rate", i18n("Sample Rate"), TQVariant::Int);
setSuffix(item, i18n(" Hz"));
item = addItemInfo(group, "UpperBitrate", i18n("Upper Bitrate"),
TQVariant::Int);
setSuffix(item, i18n(" kbps"));
item = addItemInfo(group, "LowerBitrate", i18n("Lower Bitrate"),
TQVariant::Int);
setSuffix(item, i18n(" kbps"));
item = addItemInfo(group, "NominalBitrate", i18n("Nominal Bitrate"),
TQVariant::Int);
setSuffix(item, i18n(" kbps"));
item = addItemInfo(group, "Bitrate", i18n("Average Bitrate"),
TQVariant::Int);
setAttributes(item, KFileMimeTypeInfo::Averaged);
setHint(item, KFileMimeTypeInfo::Bitrate);
setSuffix(item, i18n( " kbps"));
item = addItemInfo(group, "Length", i18n("Length"), TQVariant::Int);
setAttributes(item, KFileMimeTypeInfo::Cummulative);
setUnit(item, KFileMimeTypeInfo::Seconds);
}
bool KOggPlugin::readInfo( KFileMetaInfo& info, uint what )
{
// parts of this code taken from ogginfo.c of the vorbis-tools v1.0rc2
FILE *fp;
OggVorbis_File vf;
int rc,i;
vorbis_comment *vc;
vorbis_info *vi;
bool readComment = false;
bool readTech = false;
if (what & (KFileMetaInfo::Fastest |
KFileMetaInfo::DontCare |
KFileMetaInfo::ContentInfo)) readComment = true;
if (what & (KFileMetaInfo::Fastest |
KFileMetaInfo::DontCare |
KFileMetaInfo::TechnicalInfo)) readTech = true;
memset(&vf, 0, sizeof(OggVorbis_File));
if ( info.path().isEmpty() ) // remote file
return false;
fp = fopen(TQFile::encodeName(info.path()),"rb");
if (!fp)
{
kdDebug(7034) << "Unable to open " << TQFile::encodeName(info.path()).data() << endl;
return false;
}
rc = ov_open(fp,&vf,NULL,0);
if (rc < 0)
{
kdDebug(7034) << "Unable to understand " << TQFile::encodeName(info.path()).data()
<< ", errorcode=" << rc << endl;
return false;
}
// info.insert(KFileMetaInfoItem("Vendor", i18n("Vendor"),
// TQVariant(TQString(vi->vendor))));
// get the vorbis comments
if (readComment)
{
vc = ov_comment(&vf,-1);
KFileMetaInfoGroup commentGroup = appendGroup(info, "Comment");
for (i=0; i < vc->comments; i++)
{
kdDebug(7034) << vc->user_comments[i] << endl;
TQStringList split = TQStringList::split("=", TQString::fromUtf8(vc->user_comments[i]));
split[0] = split[0].lower();
split[0][0] = split[0][0].upper();
// we have to be sure that the i18n() string always has the same
// case. Oh, and is UTF8 ok here?
appendItem(commentGroup, split[0], split[1]);
}
}
if (readTech)
{
KFileMetaInfoGroup techgroup = appendGroup(info, "Technical");
// get other information about the file
vi = ov_info(&vf,-1);
if (vi)
{
appendItem(techgroup, "Version", int(vi->version));
appendItem(techgroup, "Channels", int(vi->channels));
appendItem(techgroup, "Sample Rate", int(vi->rate));
if (vi->bitrate_upper > 0)
appendItem(techgroup, "UpperBitrate",
int(vi->bitrate_upper+500)/1000);
if (vi->bitrate_lower > 0)
appendItem(techgroup, "LowerBitrate",
int(vi->bitrate_lower+500)/1000);
if (vi->bitrate_nominal > 0)
appendItem(techgroup, "NominalBitrate",
int(vi->bitrate_nominal+500)/1000);
if (ov_bitrate(&vf,-1) > 0)
appendItem(techgroup, "Bitrate", int(ov_bitrate(&vf,-1)+500)/1000);
}
appendItem(techgroup, "Length", int(ov_time_total(&vf,-1)));
}
ov_clear(&vf);
return true;
}
bool KOggPlugin::writeInfo(const KFileMetaInfo& info) const
{
// todo: add writing support
FILE* infile;
infile = fopen(TQFile::encodeName(info.path()), "r");
if (!infile)
{
kdDebug(7034) << "couldn't open " << info.path() << endl;
return false;
}
vcedit_state *state=vcedit_new_state();
if ( vcedit_open(state, infile)==-1 )
{
kdDebug(7034) << "error in vcedit_open for " << info.path() << endl;
return false;
}
struct vorbis_comment* oc = vcedit_comments(state);
struct vorbis_comment* vc = state->vc;
if(vc) vorbis_comment_clear(vc);
if (oc && oc->vendor)
{
vc->vendor = strdup(oc->vendor);
}
else
{
vc->vendor = strdup("");
}
KFileMetaInfoGroup group = info["Comment"];
TQStringList keys = group.keys();
TQStringList::Iterator it;
for (it = keys.begin(); it!=keys.end(); ++it)
{
KFileMetaInfoItem item = group[*it];
if (!item.isEditable() || !(item.type()==TQVariant::String) )
continue;
TQCString key = item.key().upper().utf8();
if (item.value().canCast(TQVariant::String))
{
TQCString value = item.value().toString().utf8();
kdDebug(7034) << " writing tag " << key << "=" << value << endl;
vorbis_comment_add_tag(vc,
const_cast<char*>(static_cast<const char*>(key)),
const_cast<char*>(static_cast<const char*>(value)));
}
else
kdWarning(7034) << "ignoring " << key << endl;
}
TQString filename;
TQFileInfo fileinfo(info.path());
// follow symlinks
if (fileinfo.isSymLink())
filename = fileinfo.readLink();
else
filename = info.path();
// nothing in TQt or KDE to get the mode as an int?
struct stat s;
stat(TQFile::encodeName(filename), &s);
KSaveFile sf(filename, s.st_mode);
FILE* outfile = sf.fstream();
if ( sf.status() || !outfile)
{
kdDebug(7034) << "couldn't create temp file\n";
vcedit_clear(state); // frees comment entries and vendor
sf.abort();
if (vc->vendor) free(vc->vendor);
vc->vendor = 0;
return false;
}
vcedit_write(state,outfile); // calls vcedit_clear() itself so we don't free anything
if (vc->vendor) free(vc->vendor);
vc->vendor = 0;
fclose(infile);
sf.close();
return true;
}
TQValidator* KOggPlugin::createValidator( const TQString&,
const TQString &, const TQString &,
TQObject* parent, const char* name) const {
return new TQRegExpValidator(TQRegExp(".*"), parent, name);
}
#include "tdefile_ogg.moc"