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.
k3b/libk3b/projects/audiocd/k3baudiodoc.cpp

1128 lines
33 KiB

/*
*
* $Id: k3baudiodoc.cpp 619556 2007-01-03 17:38:12Z trueg $
* Copyright (C) 2003 Sebastian Trueg <trueg@k3b.org>
*
* This file is part of the K3b project.
* Copyright (C) 1998-2007 Sebastian Trueg <trueg@k3b.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; either version 2 of the License, or
* (at your option) any later version.
* See the file "COPYING" for the exact licensing terms.
*/
#include <k3bglobals.h>
#include "k3baudiodoc.h"
#include "k3baudiotrack.h"
#include "k3baudiojob.h"
#include "k3baudiofile.h"
#include "k3baudiozerodata.h"
#include "k3baudiocdtracksource.h"
#include <k3bcuefileparser.h>
#include <k3bcdtextvalidator.h>
#include <k3bcore.h>
#include <k3baudiodecoder.h>
// QT-includes
#include <tqstring.h>
#include <tqstringlist.h>
#include <tqfile.h>
#include <tqfileinfo.h>
#include <tqdatastream.h>
#include <tqdir.h>
#include <tqdom.h>
#include <tqdatetime.h>
#include <tqtextstream.h>
#include <tqsemaphore.h>
// KDE-includes
#include <kprocess.h>
#include <kurl.h>
#include <kapplication.h>
#include <kmessagebox.h>
#include <kconfig.h>
#include <klocale.h>
#include <kstandarddirs.h>
#include <kio/global.h>
#include <kdebug.h>
#include <iostream>
class K3bAudioDoc::Private
{
public:
Private() {
cdTextValidator = new K3bCdTextValidator();
}
~Private() {
delete cdTextValidator;
}
K3bCdTextValidator* cdTextValidator;
};
K3bAudioDoc::K3bAudioDoc( TQObject* parent )
: K3bDoc( parent ),
m_firstTrack(0),
m_lastTrack(0)
{
d = new Private;
m_docType = AUDIO;
}
K3bAudioDoc::~K3bAudioDoc()
{
// delete all tracks
int i = 1;
int cnt = numOfTracks();
while( m_firstTrack ) {
kdDebug() << "(K3bAudioDoc::~K3bAudioDoc) deleting track " << i << " of " << cnt << endl;
delete m_firstTrack->take();
kdDebug() << "(K3bAudioDoc::~K3bAudioDoc) deleted." << endl;
++i;
}
delete d;
}
bool K3bAudioDoc::newDocument()
{
// delete all tracks
while( m_firstTrack )
delete m_firstTrack->take();
m_normalize = false;
m_hideFirstTrack = false;
m_cdText = false;
m_cdTextData.clear();
m_audioRippingParanoiaMode = 0;
m_audioRippingRetries = 5;
m_audioRippingIgnoreReadErrors = true;
return K3bDoc::newDocument();
}
TQString K3bAudioDoc::name() const
{
if( !m_cdTextData.title().isEmpty() )
return m_cdTextData.title();
else
return K3bDoc::name();
}
K3bAudioTrack* K3bAudioDoc::firstTrack() const
{
return m_firstTrack;
}
K3bAudioTrack* K3bAudioDoc::lastTrack() const
{
return m_lastTrack;
}
// this one is called by K3bAudioTrack to update the list
void K3bAudioDoc::setFirstTrack( K3bAudioTrack* track )
{
m_firstTrack = track;
}
// this one is called by K3bAudioTrack to update the list
void K3bAudioDoc::setLastTrack( K3bAudioTrack* track )
{
m_lastTrack = track;
}
KIO::filesize_t K3bAudioDoc::size() const
{
// This is not really correct but what the user expects ;)
return length().mode1Bytes();
}
K3b::Msf K3bAudioDoc::length() const
{
K3b::Msf length = 0;
K3bAudioTrack* track = m_firstTrack;
while( track ) {
length += track->length();
track = track->next();
}
return length;
}
void K3bAudioDoc::addUrls( const KURL::List& urls )
{
// make sure we add them at the end even if urls are in the queue
addTracks( urls, 99 );
}
void K3bAudioDoc::addTracks( const KURL::List& urls, uint position )
{
KURL::List allUrls = extractUrlList( K3b::convertToLocalUrls(urls) );
KURL::List::iterator end( allUrls.end());
for( KURL::List::iterator it = allUrls.begin(); it != end; it++, position++ ) {
KURL& url = *it;
if( url.path().right(3).lower() == "cue" ) {
// try adding a cue file
if( K3bAudioTrack* newAfter = importCueFile( url.path(), getTrack(position) ) ) {
position = newAfter->trackNumber();
continue;
}
}
if( K3bAudioTrack* track = createTrack( url ) ) {
addTrack( track, position );
K3bAudioDecoder* dec = static_cast<K3bAudioFile*>( track->firstSource() )->decoder();
track->setTitle( dec->metaInfo( K3bAudioDecoder::META_TITLE ) );
track->setArtist( dec->metaInfo( K3bAudioDecoder::META_ARTIST ) );
track->setSongwriter( dec->metaInfo( K3bAudioDecoder::META_SONGWRITER ) );
track->setComposer( dec->metaInfo( K3bAudioDecoder::META_COMPOSER ) );
track->setCdTextMessage( dec->metaInfo( K3bAudioDecoder::META_COMMENT ) );
}
}
emit changed();
informAboutNotFoundFiles();
}
KURL::List K3bAudioDoc::extractUrlList( const KURL::List& urls )
{
KURL::List allUrls = urls;
KURL::List urlsFromPlaylist;
KURL::List::iterator it = allUrls.begin();
while( it != allUrls.end() ) {
const KURL& url = *it;
TQFileInfo fi( url.path() );
if( !url.isLocalFile() ) {
kdDebug() << url.path() << " no local file" << endl;
it = allUrls.remove( it );
m_notFoundFiles.append( url );
}
else if( !fi.exists() ) {
it = allUrls.remove( it );
kdDebug() << url.path() << " not found" << endl;
m_notFoundFiles.append( url );
}
else if( fi.isDir() ) {
it = allUrls.remove( it );
// add all files in the dir
TQDir dir(fi.filePath());
TQStringList entries = dir.entryList( TQDir::Files );
KURL::List::iterator oldIt = it;
// add all files into the list after the current item
for( TQStringList::iterator dirIt = entries.begin();
dirIt != entries.end(); ++dirIt )
it = allUrls.insert( oldIt, KURL::fromPathOrURL( dir.absPath() + "/" + *dirIt ) );
}
else if( readPlaylistFile( url, urlsFromPlaylist ) ) {
it = allUrls.remove( it );
KURL::List::iterator oldIt = it;
// add all files into the list after the current item
for( KURL::List::iterator dirIt = urlsFromPlaylist.begin();
dirIt != urlsFromPlaylist.end(); ++dirIt )
it = allUrls.insert( oldIt, *dirIt );
}
else
++it;
}
return allUrls;
}
bool K3bAudioDoc::readPlaylistFile( const KURL& url, KURL::List& playlist )
{
// check if the file is a m3u playlist
// and if so add all listed files
TQFile f( url.path() );
if( !f.open( IO_ReadOnly ) )
return false;
TQTextStream t( &f );
char buf[7];
t.readRawBytes( buf, 7 );
if( TQString::fromLatin1( buf, 7 ) != "#EXTM3U" )
return false;
// skip the first line
t.readLine();
// read the file
while( !t.atEnd() ) {
TQString line = t.readLine();
if( line[0] != '#' ) {
KURL mp3url;
// relative paths
if( line[0] != '/' )
mp3url.setPath( url.directory(false) + line );
else
mp3url.setPath( line );
playlist.append( mp3url );
}
}
return true;
}
void K3bAudioDoc::addSources( K3bAudioTrack* parent,
const KURL::List& urls,
K3bAudioDataSource* sourceAfter )
{
kdDebug() << "(K3bAudioDoc::addSources( " << parent << ", "
<< urls.first().path() << ", "
<< sourceAfter << " )" << endl;
KURL::List allUrls = extractUrlList( urls );
KURL::List::const_iterator end(allUrls.end());
for( KURL::List::const_iterator it = allUrls.begin(); it != end; ++it ) {
if( K3bAudioFile* file = createAudioFile( *it ) ) {
if( sourceAfter )
file->moveAfter( sourceAfter );
else
file->moveAhead( parent->firstSource() );
sourceAfter = file;
}
}
informAboutNotFoundFiles();
kdDebug() << "(K3bAudioDoc::addSources) finished." << endl;
}
K3bAudioTrack* K3bAudioDoc::importCueFile( const TQString& cuefile, K3bAudioTrack* after, K3bAudioDecoder* decoder )
{
if( !after )
after = m_lastTrack;
kdDebug() << "(K3bAudioDoc::importCueFile( " << cuefile << ", " << after << ")" << endl;
K3bCueFileParser parser( cuefile );
if( parser.isValid() && parser.toc().contentType() == K3bDevice::AUDIO ) {
kdDebug() << "(K3bAudioDoc::importCueFile) parsed with image: " << parser.imageFilename() << endl;
// global cd-text
if( !parser.cdText().title().isEmpty() )
setTitle( parser.cdText().title() );
if( !parser.cdText().performer().isEmpty() )
setPerformer( parser.cdText().performer() );
bool reused = true;
if( !decoder )
decoder = getDecoderForUrl( KURL::fromPathOrURL(parser.imageFilename()), &reused );
if( decoder ) {
if( !reused )
decoder->analyseFile();
K3bAudioFile* newFile = 0;
unsigned int i = 0;
for( K3bDevice::Toc::const_iterator it = parser.toc().begin();
it != parser.toc().end(); ++it ) {
const K3bDevice::Track& track = *it;
newFile = new K3bAudioFile( decoder, this );
newFile->setStartOffset( track.firstSector() );
newFile->setEndOffset( track.lastSector()+1 );
K3bAudioTrack* newTrack = new K3bAudioTrack( this );
newTrack->addSource( newFile );
newTrack->moveAfter( after );
// we do not know the length of the source yet so we have to force the index value
if( track.index0() > 0 )
newTrack->m_index0Offset = track.length() - track.index0();
else
newTrack->m_index0Offset = 0;
// cd-text
newTrack->setTitle( parser.cdText()[i].title() );
newTrack->setPerformer( parser.cdText()[i].performer() );
// add the next track after this one
after = newTrack;
++i;
}
// let the last source use the data up to the end of the file
if( newFile )
newFile->setEndOffset(0);
return after;
}
}
return 0;
}
K3bAudioDecoder* K3bAudioDoc::getDecoderForUrl( const KURL& url, bool* reused )
{
K3bAudioDecoder* decoder = 0;
// check if we already have a proper decoder
if( m_decoderPresenceMap.contains( url.path() ) ) {
decoder = m_decoderPresenceMap[url.path()];
*reused = true;
}
else if( (decoder = K3bAudioDecoderFactory::createDecoder( url )) ) {
kdDebug() << "(K3bAudioDoc) using " << decoder->className()
<< " for decoding of " << url.path() << endl;
decoder->setFilename( url.path() );
*reused = false;
}
return decoder;
}
K3bAudioFile* K3bAudioDoc::createAudioFile( const KURL& url )
{
if( !TQFile::exists( url.path() ) ) {
m_notFoundFiles.append( url.path() );
kdDebug() << "(K3bAudioDoc) could not find file " << url.path() << endl;
return 0;
}
bool reused;
K3bAudioDecoder* decoder = getDecoderForUrl( url, &reused );
if( decoder ) {
if( !reused )
decoder->analyseFile();
return new K3bAudioFile( decoder, this );
}
else {
m_unknownFileFormatFiles.append( url.path() );
kdDebug() << "(K3bAudioDoc) unknown file type in file " << url.path() << endl;
return 0;
}
}
K3bAudioTrack* K3bAudioDoc::createTrack( const KURL& url )
{
kdDebug() << "(K3bAudioDoc::createTrack( " << url.path() << " )" << endl;
if( K3bAudioFile* file = createAudioFile( url ) ) {
K3bAudioTrack* newTrack = new K3bAudioTrack( this );
newTrack->setFirstSource( file );
return newTrack;
}
else
return 0;
}
void K3bAudioDoc::addTrack( const KURL& url, uint position )
{
addTracks( KURL::List(url), position );
}
K3bAudioTrack* K3bAudioDoc::getTrack( unsigned int trackNum )
{
K3bAudioTrack* track = m_firstTrack;
unsigned int i = 1;
while( track ) {
if( i == trackNum )
return track;
track = track->next();
++i;
}
return 0;
}
void K3bAudioDoc::addTrack( K3bAudioTrack* track, uint position )
{
kdDebug() << "(K3bAudioDoc::addTrack( " << track << ", " << position << " )" << endl;
track->m_parent = this;
if( !m_firstTrack )
m_firstTrack = m_lastTrack = track;
else if( position == 0 )
track->moveAhead( m_firstTrack );
else {
K3bAudioTrack* after = getTrack( position );
if( after )
track->moveAfter( after );
else
track->moveAfter( m_lastTrack ); // just to be sure it's anywhere...
}
emit changed();
}
void K3bAudioDoc::removeTrack( K3bAudioTrack* track )
{
delete track;
}
void K3bAudioDoc::moveTrack( K3bAudioTrack* track, K3bAudioTrack* after )
{
track->moveAfter( after );
}
TQString K3bAudioDoc::typeString() const
{
return "audio";
}
bool K3bAudioDoc::loadDocumentData( TQDomElement* root )
{
newDocument();
// we will parse the dom-tree and create a K3bAudioTrack for all entries immediately
// this should not take long and so not block the gui
TQDomNodeList nodes = root->childNodes();
for( uint i = 0; i < nodes.count(); i++ ) {
TQDomElement e = nodes.item(i).toElement();
if( e.isNull() )
return false;
if( e.nodeName() == "general" ) {
if( !readGeneralDocumentData( e ) )
return false;
}
else if( e.nodeName() == "normalize" )
setNormalize( e.text() == "yes" );
else if( e.nodeName() == "hide_first_track" )
setHideFirstTrack( e.text() == "yes" );
else if( e.nodeName() == "audio_ripping" ) {
TQDomNodeList ripNodes = e.childNodes();
for( uint j = 0; j < ripNodes.length(); j++ ) {
if( ripNodes.item(j).nodeName() == "paranoia_mode" )
setAudioRippingParanoiaMode( ripNodes.item(j).toElement().text().toInt() );
else if( ripNodes.item(j).nodeName() == "read_retries" )
setAudioRippingRetries( ripNodes.item(j).toElement().text().toInt() );
else if( ripNodes.item(j).nodeName() == "ignore_read_errors" )
setAudioRippingIgnoreReadErrors( ripNodes.item(j).toElement().text() == "yes" );
}
}
// parse cd-text
else if( e.nodeName() == "cd-text" ) {
if( !e.hasAttribute( "activated" ) )
return false;
writeCdText( e.attributeNode( "activated" ).value() == "yes" );
TQDomNodeList cdTextNodes = e.childNodes();
for( uint j = 0; j < cdTextNodes.length(); j++ ) {
if( cdTextNodes.item(j).nodeName() == "title" )
setTitle( cdTextNodes.item(j).toElement().text() );
else if( cdTextNodes.item(j).nodeName() == "artist" )
setArtist( cdTextNodes.item(j).toElement().text() );
else if( cdTextNodes.item(j).nodeName() == "arranger" )
setArranger( cdTextNodes.item(j).toElement().text() );
else if( cdTextNodes.item(j).nodeName() == "songwriter" )
setSongwriter( cdTextNodes.item(j).toElement().text() );
else if( cdTextNodes.item(j).nodeName() == "composer" )
setComposer( cdTextNodes.item(j).toElement().text() );
else if( cdTextNodes.item(j).nodeName() == "disc_id" )
setDisc_id( cdTextNodes.item(j).toElement().text() );
else if( cdTextNodes.item(j).nodeName() == "upc_ean" )
setUpc_ean( cdTextNodes.item(j).toElement().text() );
else if( cdTextNodes.item(j).nodeName() == "message" )
setCdTextMessage( cdTextNodes.item(j).toElement().text() );
}
}
else if( e.nodeName() == "contents" ) {
TQDomNodeList contentNodes = e.childNodes();
for( uint j = 0; j< contentNodes.length(); j++ ) {
TQDomElement trackElem = contentNodes.item(j).toElement();
// first of all we need a track
K3bAudioTrack* track = new K3bAudioTrack();
// backwards compatibility
// -----------------------------------------------------------------------------------------------------
TQDomAttr oldUrlAttr = trackElem.attributeNode( "url" );
if( !oldUrlAttr.isNull() ) {
if( K3bAudioFile* file =
createAudioFile( KURL::fromPathOrURL( oldUrlAttr.value() ) ) ) {
track->addSource( file );
}
}
// -----------------------------------------------------------------------------------------------------
TQDomNodeList trackNodes = trackElem.childNodes();
for( uint trackJ = 0; trackJ < trackNodes.length(); trackJ++ ) {
if( trackNodes.item(trackJ).nodeName() == "sources" ) {
TQDomNodeList sourcesNodes = trackNodes.item(trackJ).childNodes();
for( unsigned int sourcesIndex = 0; sourcesIndex < sourcesNodes.length(); sourcesIndex++ ) {
TQDomElement sourceElem = sourcesNodes.item(sourcesIndex).toElement();
if( sourceElem.nodeName() == "file" ) {
if( K3bAudioFile* file =
createAudioFile( KURL::fromPathOrURL( sourceElem.attributeNode( "url" ).value() ) ) ) {
file->setStartOffset( K3b::Msf::fromString( sourceElem.attributeNode( "start_offset" ).value() ) );
file->setEndOffset( K3b::Msf::fromString( sourceElem.attributeNode( "end_offset" ).value() ) );
track->addSource( file );
}
}
else if( sourceElem.nodeName() == "silence" ) {
K3bAudioZeroData* zero = new K3bAudioZeroData();
zero->setLength( K3b::Msf::fromString( sourceElem.attributeNode( "length" ).value() ) );
track->addSource( zero );
}
else if( sourceElem.nodeName() == "cdtrack" ) {
K3b::Msf length = K3b::Msf::fromString( sourceElem.attributeNode( "length" ).value() );
int titlenum = 0;
unsigned int discid = 0;
TQString title, artist, cdTitle, cdArtist;
TQDomNodeList cdTrackSourceNodes = sourceElem.childNodes();
for( unsigned int cdTrackSourceIndex = 0; cdTrackSourceIndex < cdTrackSourceNodes.length(); ++cdTrackSourceIndex ) {
TQDomElement cdTrackSourceItemElem = cdTrackSourceNodes.item(cdTrackSourceIndex).toElement();
if( cdTrackSourceItemElem.nodeName() == "title_number" )
titlenum = cdTrackSourceItemElem.text().toInt();
else if( cdTrackSourceItemElem.nodeName() == "disc_id" )
discid = cdTrackSourceItemElem.text().toUInt( 0, 16 );
else if( cdTrackSourceItemElem.nodeName() == "title" )
title = cdTrackSourceItemElem.text().toInt();
else if( cdTrackSourceItemElem.nodeName() == "artist" )
artist = cdTrackSourceItemElem.text().toInt();
else if( cdTrackSourceItemElem.nodeName() == "cdtitle" )
cdTitle = cdTrackSourceItemElem.text().toInt();
else if( cdTrackSourceItemElem.nodeName() == "cdartist" )
cdArtist = cdTrackSourceItemElem.text().toInt();
}
if( discid != 0 && titlenum > 0 ) {
K3bAudioCdTrackSource* cdtrack = new K3bAudioCdTrackSource( discid, length, titlenum,
artist, title,
cdArtist, cdTitle );
cdtrack->setStartOffset( K3b::Msf::fromString( sourceElem.attributeNode( "start_offset" ).value() ) );
cdtrack->setEndOffset( K3b::Msf::fromString( sourceElem.attributeNode( "end_offset" ).value() ) );
track->addSource( cdtrack );
}
else {
kdDebug() << "(K3bAudioDoc) invalid cdtrack source." << endl;
return false;
}
}
else {
kdDebug() << "(K3bAudioDoc) unknown source type: " << sourceElem.nodeName() << endl;
return false;
}
}
}
// load cd-text
else if( trackNodes.item(trackJ).nodeName() == "cd-text" ) {
TQDomNodeList cdTextNodes = trackNodes.item(trackJ).childNodes();
for( uint trackCdTextJ = 0; trackCdTextJ < cdTextNodes.length(); trackCdTextJ++ ) {
if( cdTextNodes.item(trackCdTextJ).nodeName() == "title" )
track->setTitle( cdTextNodes.item(trackCdTextJ).toElement().text() );
else if( cdTextNodes.item(trackCdTextJ).nodeName() == "artist" )
track->setArtist( cdTextNodes.item(trackCdTextJ).toElement().text() );
else if( cdTextNodes.item(trackCdTextJ).nodeName() == "arranger" )
track->setArranger( cdTextNodes.item(trackCdTextJ).toElement().text() );
else if( cdTextNodes.item(trackCdTextJ).nodeName() == "songwriter" )
track->setSongwriter( cdTextNodes.item(trackCdTextJ).toElement().text() );
else if( cdTextNodes.item(trackCdTextJ).nodeName() == "composer" )
track->setComposer( cdTextNodes.item(trackCdTextJ).toElement().text() );
else if( cdTextNodes.item(trackCdTextJ).nodeName() == "isrc" )
track->setIsrc( cdTextNodes.item(trackCdTextJ).toElement().text() );
else if( cdTextNodes.item(trackCdTextJ).nodeName() == "message" )
track->setCdTextMessage( cdTextNodes.item(trackCdTextJ).toElement().text() );
}
}
else if( trackNodes.item(trackJ).nodeName() == "index0" )
track->setIndex0( K3b::Msf::fromString( trackNodes.item(trackJ).toElement().text() ) );
// TODO: load other indices
// load options
else if( trackNodes.item(trackJ).nodeName() == "copy_protection" )
track->setCopyProtection( trackNodes.item(trackJ).toElement().text() == "yes" );
else if( trackNodes.item(trackJ).nodeName() == "pre_emphasis" )
track->setPreEmp( trackNodes.item(trackJ).toElement().text() == "yes" );
}
// add the track
if( track->numberSources() > 0 )
addTrack( track, 99 ); // append to the end // TODO improve
else {
kdDebug() << "(K3bAudioDoc) no sources. deleting track " << track << endl;
delete track;
}
}
}
}
informAboutNotFoundFiles();
setModified(false);
return true;
}
bool K3bAudioDoc::saveDocumentData( TQDomElement* docElem )
{
TQDomDocument doc = docElem->ownerDocument();
saveGeneralDocumentData( docElem );
// add normalize
TQDomElement normalizeElem = doc.createElement( "normalize" );
normalizeElem.appendChild( doc.createTextNode( normalize() ? "yes" : "no" ) );
docElem->appendChild( normalizeElem );
// add hide track
TQDomElement hideFirstTrackElem = doc.createElement( "hide_first_track" );
hideFirstTrackElem.appendChild( doc.createTextNode( hideFirstTrack() ? "yes" : "no" ) );
docElem->appendChild( hideFirstTrackElem );
// save the audio cd ripping settings
// paranoia mode, read retries, and ignore read errors
// ------------------------------------------------------------
TQDomElement ripMain = doc.createElement( "audio_ripping" );
docElem->appendChild( ripMain );
TQDomElement ripElem = doc.createElement( "paranoia_mode" );
ripElem.appendChild( doc.createTextNode( TQString::number( audioRippingParanoiaMode() ) ) );
ripMain.appendChild( ripElem );
ripElem = doc.createElement( "read_retries" );
ripElem.appendChild( doc.createTextNode( TQString::number( audioRippingRetries() ) ) );
ripMain.appendChild( ripElem );
ripElem = doc.createElement( "ignore_read_errors" );
ripElem.appendChild( doc.createTextNode( audioRippingIgnoreReadErrors() ? "yes" : "no" ) );
ripMain.appendChild( ripElem );
// ------------------------------------------------------------
// save disc cd-text
// -------------------------------------------------------------
TQDomElement cdTextMain = doc.createElement( "cd-text" );
cdTextMain.setAttribute( "activated", cdText() ? "yes" : "no" );
TQDomElement cdTextElem = doc.createElement( "title" );
cdTextElem.appendChild( doc.createTextNode( (title())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "artist" );
cdTextElem.appendChild( doc.createTextNode( (artist())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "arranger" );
cdTextElem.appendChild( doc.createTextNode( (arranger())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "songwriter" );
cdTextElem.appendChild( doc.createTextNode( (songwriter())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "composer" );
cdTextElem.appendChild( doc.createTextNode( composer()) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "disc_id" );
cdTextElem.appendChild( doc.createTextNode( (disc_id())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "upc_ean" );
cdTextElem.appendChild( doc.createTextNode( (upc_ean())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "message" );
cdTextElem.appendChild( doc.createTextNode( (cdTextMessage())) );
cdTextMain.appendChild( cdTextElem );
docElem->appendChild( cdTextMain );
// -------------------------------------------------------------
// save the tracks
// -------------------------------------------------------------
TQDomElement contentsElem = doc.createElement( "contents" );
for( K3bAudioTrack* track = firstTrack(); track != 0; track = track->next() ) {
TQDomElement trackElem = doc.createElement( "track" );
// add sources
TQDomElement sourcesParent = doc.createElement( "sources" );
for( K3bAudioDataSource* source = track->firstSource(); source; source = source->next() ) {
// TODO: save a source element with a type attribute and start- and endoffset
// then distict between the different source types.
if( K3bAudioFile* file = dynamic_cast<K3bAudioFile*>(source) ) {
TQDomElement sourceElem = doc.createElement( "file" );
sourceElem.setAttribute( "url", file->filename() );
sourceElem.setAttribute( "start_offset", file->startOffset().toString() );
sourceElem.setAttribute( "end_offset", file->endOffset().toString() );
sourcesParent.appendChild( sourceElem );
}
else if( K3bAudioZeroData* zero = dynamic_cast<K3bAudioZeroData*>(source) ) {
TQDomElement sourceElem = doc.createElement( "silence" );
sourceElem.setAttribute( "length", zero->length().toString() );
sourcesParent.appendChild( sourceElem );
}
else if( K3bAudioCdTrackSource* cdTrack = dynamic_cast<K3bAudioCdTrackSource*>(source) ) {
TQDomElement sourceElem = doc.createElement( "cdtrack" );
sourceElem.setAttribute( "length", cdTrack->originalLength().toString() );
sourceElem.setAttribute( "start_offset", cdTrack->startOffset().toString() );
sourceElem.setAttribute( "end_offset", cdTrack->endOffset().toString() );
TQDomElement subElem = doc.createElement( "title_number" );
subElem.appendChild( doc.createTextNode( TQString::number(cdTrack->cdTrackNumber()) ) );
sourceElem.appendChild( subElem );
subElem = doc.createElement( "disc_id" );
subElem.appendChild( doc.createTextNode( TQString::number(cdTrack->discId(), 16) ) );
sourceElem.appendChild( subElem );
subElem = doc.createElement( "title" );
subElem.appendChild( doc.createTextNode( cdTrack->metaInfo().titles[cdTrack->cdTrackNumber()-1] ) );
sourceElem.appendChild( subElem );
subElem = doc.createElement( "artist" );
subElem.appendChild( doc.createTextNode( cdTrack->metaInfo().artists[cdTrack->cdTrackNumber()-1] ) );
sourceElem.appendChild( subElem );
subElem = doc.createElement( "cdtitle" );
subElem.appendChild( doc.createTextNode( cdTrack->metaInfo().cdTitle ) );
sourceElem.appendChild( subElem );
subElem = doc.createElement( "cdartist" );
subElem.appendChild( doc.createTextNode( cdTrack->metaInfo().cdArtist ) );
sourceElem.appendChild( subElem );
sourcesParent.appendChild( sourceElem );
}
else {
kdError() << "(K3bAudioDoc) saving sources other than file or zero not supported yet." << endl;
return false;
}
}
trackElem.appendChild( sourcesParent );
// index 0
TQDomElement index0Elem = doc.createElement( "index0" );
index0Elem.appendChild( doc.createTextNode( track->index0().toString() ) );
trackElem.appendChild( index0Elem );
// TODO: other indices
// add cd-text
cdTextMain = doc.createElement( "cd-text" );
cdTextElem = doc.createElement( "title" );
cdTextElem.appendChild( doc.createTextNode( (track->title())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "artist" );
cdTextElem.appendChild( doc.createTextNode( (track->artist())) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "arranger" );
cdTextElem.appendChild( doc.createTextNode( (track->arranger()) ) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "songwriter" );
cdTextElem.appendChild( doc.createTextNode( (track->songwriter()) ) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "composer" );
cdTextElem.appendChild( doc.createTextNode( (track->composer()) ) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "isrc" );
cdTextElem.appendChild( doc.createTextNode( ( track->isrc()) ) );
cdTextMain.appendChild( cdTextElem );
cdTextElem = doc.createElement( "message" );
cdTextElem.appendChild( doc.createTextNode( (track->cdTextMessage()) ) );
cdTextMain.appendChild( cdTextElem );
trackElem.appendChild( cdTextMain );
// add copy protection
TQDomElement copyElem = doc.createElement( "copy_protection" );
copyElem.appendChild( doc.createTextNode( track->copyProtection() ? "yes" : "no" ) );
trackElem.appendChild( copyElem );
// add pre emphasis
copyElem = doc.createElement( "pre_emphasis" );
copyElem.appendChild( doc.createTextNode( track->preEmp() ? "yes" : "no" ) );
trackElem.appendChild( copyElem );
contentsElem.appendChild( trackElem );
}
// -------------------------------------------------------------
docElem->appendChild( contentsElem );
return true;
}
int K3bAudioDoc::numOfTracks() const
{
return ( m_lastTrack ? m_lastTrack->trackNumber() : 0 );
}
K3bBurnJob* K3bAudioDoc::newBurnJob( K3bJobHandler* hdl, TQObject* parent )
{
return new K3bAudioJob( this, hdl, parent );
}
void K3bAudioDoc::informAboutNotFoundFiles()
{
if( !m_notFoundFiles.isEmpty() ) {
TQStringList l;
for( KURL::List::const_iterator it = m_notFoundFiles.begin();
it != m_notFoundFiles.end(); ++it )
l.append( (*it).path() );
KMessageBox::informationList( TQT_TQWIDGET(tqApp->activeWindow()),
i18n("Could not find the following files:"),
l,
i18n("Not Found") );
m_notFoundFiles.clear();
}
if( !m_unknownFileFormatFiles.isEmpty() ) {
TQStringList l;
for( KURL::List::const_iterator it = m_unknownFileFormatFiles.begin();
it != m_unknownFileFormatFiles.end(); ++it )
l.append( (*it).path() );
KMessageBox::informationList( TQT_TQWIDGET(tqApp->activeWindow()),
i18n("<p>Unable to handle the following files due to an unsupported format:"
"<p>You may manually convert these audio files to wave using another "
"application supporting the audio format and then add the wave files "
"to the K3b project."),
l,
i18n("Unsupported Format") );
m_unknownFileFormatFiles.clear();
}
}
void K3bAudioDoc::removeCorruptTracks()
{
// K3bAudioTrack* track = m_tracks->first();
// while( track ) {
// if( track->status() != 0 ) {
// removeTrack(track);
// track = m_tracks->current();
// }
// else
// track = m_tracks->next();
// }
}
void K3bAudioDoc::slotTrackChanged( K3bAudioTrack* track )
{
kdDebug() << "(K3bAudioDoc::slotTrackChanged " << track << endl;
setModified( true );
// if the track is empty now we simply delete it
if( track->firstSource() ) {
emit trackChanged(track);
emit changed();
}
else {
kdDebug() << "(K3bAudioDoc::slotTrackChanged) track " << track << " empty. Deleting." << endl;
delete track; // this will emit the proper signal
}
kdDebug() << "(K3bAudioDoc::slotTrackChanged done" << track << endl;
}
void K3bAudioDoc::slotTrackRemoved( K3bAudioTrack* track )
{
setModified( true );
emit trackRemoved(track);
emit changed();
}
void K3bAudioDoc::increaseDecoderUsage( K3bAudioDecoder* decoder )
{
kdDebug() << "(K3bAudioDoc::increaseDecoderUsage)" << endl;
if( !m_decoderUsageCounterMap.contains( decoder ) ) {
m_decoderUsageCounterMap[decoder] = 1;
m_decoderPresenceMap[decoder->filename()] = decoder;
}
else
m_decoderUsageCounterMap[decoder]++;
kdDebug() << "(K3bAudioDoc::increaseDecoderUsage) finished" << endl;
}
void K3bAudioDoc::decreaseDecoderUsage( K3bAudioDecoder* decoder )
{
m_decoderUsageCounterMap[decoder]--;
if( m_decoderUsageCounterMap[decoder] <= 0 ) {
m_decoderUsageCounterMap.erase(decoder);
m_decoderPresenceMap.erase(decoder->filename());
delete decoder;
}
}
K3bDevice::CdText K3bAudioDoc::cdTextData() const
{
K3bDevice::CdText text( m_cdTextData );
text.reserve( numOfTracks() );
K3bAudioTrack* track = firstTrack();
while( track ) {
text.append( track->cdText() );
track = track->next();
}
return text;
}
K3bDevice::Toc K3bAudioDoc::toToc() const
{
K3bDevice::Toc toc;
// FIXME: add MCN
K3bAudioTrack* track = firstTrack();
K3b::Msf pos = 0;
while( track ) {
toc.append( track->toCdTrack() );
track = track->next();
}
return toc;
}
void K3bAudioDoc::setTitle( const TQString& v )
{
m_cdTextData.setTitle( v );
emit changed();
}
void K3bAudioDoc::setArtist( const TQString& v )
{
setPerformer( v );
}
void K3bAudioDoc::setPerformer( const TQString& v )
{
TQString s( v );
d->cdTextValidator->fixup( s );
m_cdTextData.setPerformer( s );
emit changed();
}
void K3bAudioDoc::setDisc_id( const TQString& v )
{
TQString s( v );
d->cdTextValidator->fixup( s );
m_cdTextData.setDiscId( s );
emit changed();
}
void K3bAudioDoc::setArranger( const TQString& v )
{
TQString s( v );
d->cdTextValidator->fixup( s );
m_cdTextData.setArranger( s );
emit changed();
}
void K3bAudioDoc::setSongwriter( const TQString& v )
{
TQString s( v );
d->cdTextValidator->fixup( s );
m_cdTextData.setSongwriter( s );
emit changed();
}
void K3bAudioDoc::setComposer( const TQString& v )
{
TQString s( v );
d->cdTextValidator->fixup( s );
m_cdTextData.setComposer( s );
emit changed();
}
void K3bAudioDoc::setUpc_ean( const TQString& v )
{
TQString s( v );
d->cdTextValidator->fixup( s );
m_cdTextData.setUpcEan( s );
emit changed();
}
void K3bAudioDoc::setCdTextMessage( const TQString& v )
{
TQString s( v );
d->cdTextValidator->fixup( s );
m_cdTextData.setMessage( s );
emit changed();
}
#include "k3baudiodoc.moc"