Logo Search packages:      
Sourcecode: k3b version File versions

k3baudiodoc.cpp

/*
 *
 * $Id: k3baudiodoc.cpp,v 1.18.2.4 2004/09/27 07:18:03 trueg Exp $
 * Copyright (C) 2003 Sebastian Trueg <trueg@k3b.org>
 *
 * This file is part of the K3b project.
 * Copyright (C) 1998-2004 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 "k3baudioview.h"
#include "k3baudiotrack.h"
#include "k3baudioburndialog.h"
#include "k3baudiojob.h"

#include <songdb/k3bsong.h>
#include <songdb/k3bsongmanager.h>
#include <k3bthread.h>
#include <k3bthreadjob.h>
#include <k3bcore.h>
#include <k3bpluginmanager.h>
#include <k3baudiodecoder.h>


// QT-includes
#include <qstring.h>
#include <qstringlist.h>
#include <qfile.h>
#include <qfileinfo.h>
#include <qdatastream.h>
#include <qdir.h>
#include <qdom.h>
#include <qdatetime.h>
#include <qtimer.h>
#include <qtextstream.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>


// this simple thread is just used to asynchronously determine the
// length and integrety of the tracks
class K3bAudioDoc::AudioTrackStatusThread : public K3bThread
{
public:
  AudioTrackStatusThread()
    : K3bThread(),
      m_track(0) {
  }

  void analyseTrack( K3bAudioTrack* track ) {
    m_track = track;
    start();
  }

  K3bAudioTrack* track() const {
    return m_track;
  }

protected:
  void run() {
    kdDebug() << "(AudioTrackStatusThread) run" << endl;
    if( m_track->module()->analyseFile() ) {
      m_track->setStatus( 0 );

      // 
      // Do not overwrite loaded cd-text data
      // FIXME: this is a bad solution :(
      //
      if( m_track->cdText().isEmpty() ) {
      
      // first search the songdb
      K3bSong *song = K3bSongManager::instance()->findSong( m_track->path() );
      if( song != 0 ){
        m_track->setArtist( song->getArtist() );
        m_track->setTitle( song->getTitle() );
      }
      else {
        // no song found, try the module
        m_track->setTitle( m_track->module()->metaInfo( K3bAudioDecoder::META_TITLE ) );
        m_track->setPerformer( m_track->module()->metaInfo( K3bAudioDecoder::META_ARTIST ) );
      }
      
      m_track->setComposer( m_track->module()->metaInfo( K3bAudioDecoder::META_COMPOSER ) );
      m_track->setSongwriter( m_track->module()->metaInfo( K3bAudioDecoder::META_SONGWRITER ) );
      m_track->setCdTextMessage( m_track->module()->metaInfo( K3bAudioDecoder::META_COMMENT ) );
      }
    }
    else
      m_track->setStatus( -1 );
    kdDebug() << "(AudioTrackStatusThread) finished" << endl;
    emitFinished(true);
  }

private:
  K3bAudioTrack* m_track;
};



K3bAudioDoc::K3bAudioDoc( QObject* parent )
  : K3bDoc( parent )
{
  m_tracks = 0L;
  m_cdText = false;
  m_padding = true;  // padding is enabled forever since there is no use in disabling it!

  m_docType = AUDIO;

  m_urlAddingTimer = new QTimer( this );
  connect( m_urlAddingTimer, SIGNAL(timeout()), this, SLOT(slotWorkUrlQueue()) );

  m_trackStatusThread = new AudioTrackStatusThread();
  m_trackMetaInfoJob = new K3bThreadJob( this );
  m_trackMetaInfoJob->setThread( m_trackStatusThread );
  connect( m_trackMetaInfoJob, SIGNAL(finished(bool)),
         this, SLOT(slotDetermineTrackStatus()) );
  connect( m_trackMetaInfoJob, SIGNAL(finished(bool)),
         this, SIGNAL(changed()) );

  // FIXME: remove the newTracks() signal and replace it with the changed signal
  connect( this, SIGNAL(newTracks()), this, SIGNAL(changed()) );
  connect( this, SIGNAL(trackRemoved(K3bAudioTrack*)), this, SIGNAL(changed()) );
}

K3bAudioDoc::~K3bAudioDoc()
{
  if( m_tracks )
    m_tracks->setAutoDelete( true );

  delete m_tracks;
  delete m_trackStatusThread;
}

00153 bool K3bAudioDoc::newDocument()
{
  if( m_tracks ) {
    while( m_tracks->first() )
      removeTrack( m_tracks->first() );
  }
  else
    m_tracks = new QPtrList<K3bAudioTrack>;
  m_tracks->setAutoDelete(false);

  return K3bDoc::newDocument();
}



00168 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;
  for( QPtrListIterator<K3bAudioTrack> it(*m_tracks); it.current(); ++it ) {
    length += it.current()->length() + it.current()->pregap();
  }   

  return length;
}


00186 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 )
{
  for( KURL::List::ConstIterator it = urls.begin(); it != urls.end(); it++ ) {
    urlsToAdd.enqueue( new PrivateUrlToAdd( *it, position++ ) );
  }

  m_urlAddingTimer->start(0);
}

00202 void K3bAudioDoc::slotWorkUrlQueue()
{
  if( !urlsToAdd.isEmpty() ) {
    PrivateUrlToAdd* item = urlsToAdd.dequeue();
    lastAddedPosition = item->position;

    // append at the end by default
    if( lastAddedPosition > m_tracks->count() )
      lastAddedPosition = m_tracks->count();

    if( !item->url.isLocalFile() ) {
      kdDebug() << item->url.path() << " no local file" << endl;
      m_notFoundFiles.append( item->url.path() );
      delete item;
      return;
    }

    QFileInfo fi( item->url.path() );
    if( !fi.exists() ) {
      m_notFoundFiles.append( item->url.path() );
      delete item;
      return;
    }

    if( fi.isDir() ) {
      // add all files in the dir
      QDir dir(fi.filePath());
      QStringList entries = dir.entryList( QDir::Files );
      KURL::List urls;
      for( QStringList::iterator it = entries.begin();
         it != entries.end(); ++it )
      urls.append( KURL::fromPathOrURL( dir.absPath() + "/" + *it ) );

      addTracks( urls, lastAddedPosition++ );

      delete item;
      return;
    }

    if( !readM3uFile( item->url, lastAddedPosition ) )
      if( K3bAudioTrack* newTrack = createTrack( item->url ) ) {
        addTrack( newTrack, lastAddedPosition );
      slotDetermineTrackStatus();
      }

    delete item;

    emit newTracks();
  }

  else {
    m_urlAddingTimer->stop();

    emit newTracks();

    informAboutNotFoundFiles();
  }
}


bool K3bAudioDoc::readM3uFile( const KURL& url, int pos )
{
  // check if the file is a m3u playlist
  // and if so add all listed files

  QFile f( url.path() );
  if( !f.open( IO_ReadOnly ) )
    return false;

  QTextStream t( &f );
  char buf[7];
  t.readRawBytes( buf, 7 );
  if( QString::fromLatin1( buf, 7 ) != "#EXTM3U" )
    return false;

  // skip the first line
  t.readLine();

  // read the file
  while( !t.atEnd() ) {
    QString line = t.readLine();
    if( line[0] != '#' ) {
      KURL mp3url;
      // relative paths
      if( line[0] != '/' )
        mp3url.setPath( url.directory(false) + line );
      else
        mp3url.setPath( line );
      urlsToAdd.enqueue( new PrivateUrlToAdd( mp3url , pos++ ) );
    }
  }

  m_urlAddingTimer->start(0);
  return true;
}


K3bAudioTrack* K3bAudioDoc::createTrack( const KURL& url )
{
  QPtrList<K3bPluginFactory> fl = k3bpluginmanager->factories( "AudioDecoder" );

  //
  // This is an evil hack to force the wave decoder to be checked first since
  // the mad decoder's canDecode method is not very reliable. :(
  //
  for( K3bPluginFactory* factory = fl.first();
       factory; factory = fl.next() ) {
    if( factory->isA( "K3bWaveDecoderFactory" ) ) {
      fl.prepend( fl.take() );
      break;
    }
  }
  
  for( QPtrListIterator<K3bPluginFactory> it( fl );
       it.current(); ++it ) {
    if( ((K3bAudioDecoderFactory*)it.current())->canDecode( url ) ) {
      kdDebug() << "(K3bAudioDoc) using " << it.current()->className() << " for decoding of " << url.path() << endl;
      K3bAudioTrack* newTrack =  new K3bAudioTrack( m_tracks, url.path() );
      connect( newTrack, SIGNAL(changed()), this, SLOT(slotTrackChanged()) );
      newTrack->setModule( (K3bAudioDecoder*)((K3bAudioDecoderFactory*)it.current())->createPlugin() );
      return newTrack;
    }
  }

  m_unknownFileFormatFiles.append( url.path() );
  return 0;
}


void K3bAudioDoc::addTrack(const KURL& url, uint position )
{
  urlsToAdd.enqueue( new PrivateUrlToAdd( url, position ) );

  m_urlAddingTimer->start(0);
}



00340 void K3bAudioDoc::addTrack( K3bAudioTrack* track, uint position )
{
  if( m_tracks->count() >= 99 ) {
    kdDebug() << "(K3bAudioDoc) Red Book only allows 99 tracks." << endl;
    // TODO: show some messagebox
    delete track;
    return;
  }

  lastAddedPosition = position;

  if( !m_tracks->insert( position, track ) ) {
    lastAddedPosition = m_tracks->count();
    m_tracks->insert( m_tracks->count(), track );
  }

  emit newTracks();
  
  setModified( true );
}


void K3bAudioDoc::removeTrack( K3bAudioTrack* track )
{
  if( !track ) {
    return;
  }
      
  // set the current item to track
  if( m_tracks->findRef( track ) >= 0 ) {
    // take the current item
    track = m_tracks->take();

    // if the AudioTrackStatusThread currently works this file we kill and restart it
    if( m_trackStatusThread->track() == track && m_trackStatusThread->running() )
      m_trackMetaInfoJob->cancel(); // this will emit a finished signal
      
    // emit signal before deleting the track to avoid crashes
    // when the view tries to call some of the tracks' methods
    emit trackRemoved( track );
      
    delete track;

    setModified();
  }
}

void K3bAudioDoc::moveTrack( const K3bAudioTrack* track, const K3bAudioTrack* after )
{
  if( track == after )
    return;

  // set the current item to track
  m_tracks->findRef( track );
  // take the current item
  track = m_tracks->take();

  // if after == 0 findRef returnes -1
  int pos = m_tracks->findRef( after );
  m_tracks->insert( pos+1, track );

  setModified();

  emit changed();
}


00407 K3bView* K3bAudioDoc::newView( QWidget* parent )
{
  return new K3bAudioView( this, parent );
}


00413 QString K3bAudioDoc::documentType() const
{
  return "audio";
}


00419 bool K3bAudioDoc::loadDocumentData( QDomElement* 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

  QDomNodeList nodes = root->childNodes();

  for( uint i = 0; i < nodes.count(); i++ ) {

    QDomElement 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" );
    

    // parse cd-text
    else if( e.nodeName() == "cd-text" ) {
      if( !e.hasAttribute( "activated" ) )
      return false;

      writeCdText( e.attributeNode( "activated" ).value() == "yes" );
    
      QDomNodeList 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" ) {
      
      QDomNodeList contentNodes = e.childNodes();

      for( uint j = 0; j< contentNodes.length(); j++ ) {

      // check if url is available
      QDomElement trackElem = contentNodes.item(j).toElement();

      QString url = trackElem.attributeNode( "url" ).value();
      if( !QFile::exists( url ) ) {
        m_notFoundFiles.append( url );
      }
      else {
        KURL k;
        k.setPath( url );
        if( K3bAudioTrack* track = createTrack( k ) ) {
          QDomNodeList trackNodes = trackElem.childNodes();
          
          for( uint trackJ = 0; trackJ < trackNodes.length(); trackJ++ ) {
            if( trackNodes.item(trackJ).nodeName() == "cd-text" ) {

            QDomNodeList 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() == "pregap" )
            track->setPregap( trackNodes.item(trackJ).toElement().text().toInt() );

            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" );
          }
          
          addTrack( track, m_tracks->count() );
        }
      }
      }
    }
  }

  emit newTracks();

  slotDetermineTrackStatus();

  informAboutNotFoundFiles();

  setModified(false);

  return true;
}

00557 bool K3bAudioDoc::saveDocumentData( QDomElement* docElem )
{
  QDomDocument doc = docElem->ownerDocument();
  saveGeneralDocumentData( docElem );

  // add normalize
  QDomElement normalizeElem = doc.createElement( "normalize" );
  normalizeElem.appendChild( doc.createTextNode( normalize() ? "yes" : "no" ) );
  docElem->appendChild( normalizeElem );

  // add hide track
  QDomElement hideFirstTrackElem = doc.createElement( "hide_first_track" );
  hideFirstTrackElem.appendChild( doc.createTextNode( hideFirstTrack() ? "yes" : "no" ) );
  docElem->appendChild( hideFirstTrackElem );


  // save disc cd-text
  // -------------------------------------------------------------
  QDomElement cdTextMain = doc.createElement( "cd-text" );
  cdTextMain.setAttribute( "activated", cdText() ? "yes" : "no" );
  QDomElement 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( "composer" );
  cdTextElem.appendChild( doc.createTextNode( composer()) );
  cdTextMain.appendChild( cdTextElem );

  cdTextElem = doc.createElement( "songwriter" );
  cdTextElem.appendChild( doc.createTextNode( (songwriter())) );
  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
  // -------------------------------------------------------------
  QDomElement contentsElem = doc.createElement( "contents" );

  for( K3bAudioTrack* track = first(); track != 0; track = next() ) {

    QDomElement trackElem = doc.createElement( "track" );
    trackElem.setAttribute( "url", KIO::decodeFileName(track->path()) );

    // 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 pregap
    QDomElement pregapElem = doc.createElement( "pregap" );    
    pregapElem.appendChild( doc.createTextNode( QString::number(track->pregap().totalFrames()) ) );
    trackElem.appendChild( pregapElem );

    // add copy protection
    QDomElement 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_tracks->count();
}


bool K3bAudioDoc::padding() const
{
  return m_padding;
}


K3bBurnJob* K3bAudioDoc::newBurnJob()
{
  return new K3bAudioJob( this );
}


void K3bAudioDoc::informAboutNotFoundFiles()
{
  if( !m_notFoundFiles.isEmpty() ) {
    KMessageBox::informationList( qApp->activeWindow(), i18n("Could not find the following files:"),
                          m_notFoundFiles, i18n("Not Found") );

    m_notFoundFiles.clear();
  }
  if( !m_unknownFileFormatFiles.isEmpty() ) {
    KMessageBox::informationList( qApp->activeWindow(), i18n("Unable to handle the following files due to an unsupported format:"),
                          m_unknownFileFormatFiles, i18n("Unsupported Format") );

    m_unknownFileFormatFiles.clear();
  }
}


00714 void K3bAudioDoc::loadDefaultSettings( KConfig* c )
{
  K3bDoc::loadDefaultSettings(c);

  m_cdText = c->readBoolEntry( "cd_text", false );
  m_padding = true;  // padding is always a good idea!
  m_hideFirstTrack = c->readBoolEntry( "hide_first_track", false );
  setNormalize( c->readBoolEntry( "normalize", false ) );
}


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::slotDetermineTrackStatus()
{
  kdDebug() << "(K3bAudioDoc) slotDetermineTrackStatus()" << endl;
  if( !m_trackMetaInfoJob->running() ) {
    kdDebug() << "(K3bAudioDoc) AudioTrackStatusThread not running." << endl;
    // find the next track
    for( QPtrListIterator<K3bAudioTrack> it( *m_tracks ); it.current(); ++it ) {
      if( it.current()->length() == 0 && it.current()->status() == 0 ) {
      kdDebug() << "(K3bAudioDoc) starting AudioTrackStatusThread for " << it.current()->path() << endl;
      m_trackStatusThread->analyseTrack( it.current() );
      return;
      }
    }
  }
  else
    kdDebug() << "(K3bAudioDoc) AudioTrackStatusThread running." << endl;
  kdDebug() << "(K3bAudioDoc) slotDetermineTrackStatus() end" << endl;
}


00759 K3bProjectBurnDialog* K3bAudioDoc::newBurnDialog( QWidget* parent, const char* name )
{
  return new K3bAudioBurnDialog( this, parent, name, true );
}


void K3bAudioDoc::slotTrackChanged()
{
  setModified( true );
  emit changed();
}


#include "k3baudiodoc.moc"

Generated by  Doxygen 1.6.0   Back to index