X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Fclipmanager.cpp;h=22fee7559ab503448312ff01a5e4bc725704a000;hb=a3eee7be24126f5a3458d488f44cd61c66135e17;hp=a06874543bb3d5dbded0a5ea7c970fd28497b7b6;hpb=848304cd283eb5a6c13cecc57b696e91bb92f729;p=kdenlive diff --git a/src/clipmanager.cpp b/src/clipmanager.cpp index a0687454..22fee755 100644 --- a/src/clipmanager.cpp +++ b/src/clipmanager.cpp @@ -26,6 +26,7 @@ #include "abstractclipitem.h" #include "abstractgroupitem.h" #include "titledocument.h" +#include "subprojectitem.h" #include "kthumb.h" #include @@ -35,26 +36,37 @@ #include #include #include +#include + +#include +#include +#include +#include #include #include +#include #include -ClipManager::ClipManager(KdenliveDoc *doc) : + +ClipManager::ClipManager(KdenliveDoc *doc, QGLWidget *glContext) : QObject(), + m_mainGLContext(glContext), m_audioThumbsQueue(), m_doc(doc), - m_generatingAudioId(), m_abortThumb(false), - m_closing(false) + m_closing(false), + m_abortAudioThumb(false) { m_clipIdCounter = 1; m_folderIdCounter = 1; m_modifiedTimer.setInterval(1500); - connect(&m_fileWatcher, SIGNAL(dirty(const QString &)), this, SLOT(slotClipModified(const QString &))); - connect(&m_fileWatcher, SIGNAL(deleted(const QString &)), this, SLOT(slotClipMissing(const QString &))); - connect(&m_fileWatcher, SIGNAL(created(const QString &)), this, SLOT(slotClipAvailable(const QString &))); + connect(&m_fileWatcher, SIGNAL(dirty(QString)), this, SLOT(slotClipModified(QString))); + connect(&m_fileWatcher, SIGNAL(deleted(QString)), this, SLOT(slotClipMissing(QString))); + + // Seems like a dirty signal is emitted anyways when a watched file is created, so don't react twice. + //connect(&m_fileWatcher, SIGNAL(created(QString)), this, SLOT(slotClipAvailable(QString))); connect(&m_modifiedTimer, SIGNAL(timeout()), this, SLOT(slotProcessModifiedClips())); #if KDE_IS_VERSION(4,5,0) @@ -68,15 +80,14 @@ ClipManager::~ClipManager() { m_closing = true; m_abortThumb = true; + m_abortAudioThumb = true; m_thumbsThread.waitForFinished(); + m_audioThumbsThread.waitForFinished(); m_thumbsMutex.lock(); m_requestedThumbs.clear(); - m_thumbsMutex.unlock(); m_audioThumbsQueue.clear(); - m_generatingAudioId.clear(); - m_thumbsMutex.lock(); - m_requestedThumbs.clear(); m_thumbsMutex.unlock(); + qDeleteAll(m_clipList); m_clipList.clear(); #if KDE_IS_VERSION(4,5,0) @@ -87,13 +98,16 @@ ClipManager::~ClipManager() void ClipManager::clear() { m_abortThumb = true; + m_abortAudioThumb = true; m_thumbsThread.waitForFinished(); + m_audioThumbsThread.waitForFinished(); m_thumbsMutex.lock(); m_requestedThumbs.clear(); + m_audioThumbsQueue.clear(); m_thumbsMutex.unlock(); m_abortThumb = false; + m_abortAudioThumb = false; m_folderList.clear(); - m_audioThumbsQueue.clear(); m_modifiedClips.clear(); qDeleteAll(m_clipList); m_clipList.clear(); @@ -111,7 +125,7 @@ void ClipManager::clearCache() #endif } -void ClipManager::requestThumbs(const QString id, QList frames) +void ClipManager::slotRequestThumbs(const QString &id, const QList & frames) { m_thumbsMutex.lock(); foreach (int frame, frames) { @@ -125,94 +139,260 @@ void ClipManager::requestThumbs(const QString id, QList frames) void ClipManager::stopThumbs(const QString &id) { - if (m_requestedThumbs.isEmpty() || m_closing) return; + if (m_closing || (m_requestedThumbs.isEmpty() && m_processingThumbId != id && m_audioThumbsQueue.isEmpty() && m_processingAudioThumbId != id)) return; + // Abort video thumbs for this clip m_abortThumb = true; m_thumbsThread.waitForFinished(); m_thumbsMutex.lock(); m_requestedThumbs.remove(id); + m_audioThumbsQueue.removeAll(id); m_thumbsMutex.unlock(); m_abortThumb = false; - if (!m_thumbsThread.isRunning()) { + + // Abort audio thumbs for this clip + if (m_processingAudioThumbId == id) { + m_abortAudioThumb = true; + m_audioThumbsThread.waitForFinished(); + m_abortAudioThumb = false; + } + + if (!m_thumbsThread.isRunning() && !m_requestedThumbs.isEmpty()) { m_thumbsThread = QtConcurrent::run(this, &ClipManager::slotGetThumbs); } + + if (!m_audioThumbsThread.isRunning() && !m_audioThumbsQueue.isEmpty()) { + m_audioThumbsThread = QtConcurrent::run(this, &ClipManager::slotGetAudioThumbs); + } } void ClipManager::slotGetThumbs() { - QMap::iterator i = m_requestedThumbs.begin(); - while (i != m_requestedThumbs.end() && !m_abortThumb) { - QString producerId = i.key(); + // We are in a new thread, so we need a new OpenGL context for the remainder of the function. + QGLWidget ctx(0, m_mainGLContext); + ctx.makeCurrent(); + + QMap::const_iterator i; + int max; + int done = 0; + int thumbType = 0; // 0 = timeline thumb, 1 = project clip zone thumb, 2 = clip properties thumb + + while (!m_requestedThumbs.isEmpty() && !m_abortThumb) { m_thumbsMutex.lock(); - QList values = m_requestedThumbs.values(producerId); - i = m_requestedThumbs.erase(i); + i = m_requestedThumbs.constBegin(); + m_processingThumbId = i.key(); + QList values = m_requestedThumbs.values(m_processingThumbId); + m_requestedThumbs.remove(m_processingThumbId); + if (m_processingThumbId.startsWith("?")) { + // if id starts with ?, it means the request comes from a clip property widget + thumbType = 2; + m_processingThumbId.remove(0, 1); + } + if (m_processingThumbId.startsWith("#")) { + // if id starts with #, it means the request comes from project tree + thumbType = 1; + m_processingThumbId.remove(0, 1); + } m_thumbsMutex.unlock(); qSort(values); - DocClipBase *clip = getClipById(producerId); + DocClipBase *clip = getClipById(m_processingThumbId); if (!clip) continue; + max = m_requestedThumbs.size() + values.count(); + int pos; while (!values.isEmpty() && clip->thumbProducer() && !m_abortThumb) { - clip->thumbProducer()->getThumb(values.takeFirst()); + pos = values.takeFirst(); + switch (thumbType) { + case 1: + clip->thumbProducer()->getGenericThumb(pos, SubProjectItem::itemDefaultHeight(), thumbType); + break; + case 2: + clip->thumbProducer()->getGenericThumb(pos, 180, thumbType); + break; + default: + clip->thumbProducer()->getThumb(pos); + } + done++; + if (max > 3) emit displayMessage(i18n("Loading thumbnails"), 100 * done / max); } } + m_processingThumbId.clear(); + emit displayMessage(QString(), -1); } void ClipManager::checkAudioThumbs() { if (!KdenliveSettings::audiothumbnails()) { - if (!m_generatingAudioId.isEmpty()) { - DocClipBase *clip = getClipById(m_generatingAudioId); - if (clip) clip->slotClearAudioCache(); + if (m_audioThumbsThread.isRunning()) { + m_abortAudioThumb = true; + m_thumbsMutex.lock(); + m_audioThumbsQueue.clear(); + m_thumbsMutex.unlock(); + m_audioThumbsThread.waitForFinished(); + m_abortAudioThumb = false; } - m_audioThumbsQueue.clear(); - m_generatingAudioId.clear(); return; } - for (int i = 0; i < m_clipList.count(); i++) { - m_audioThumbsQueue.append(m_clipList.at(i)->getId()); + m_thumbsMutex.lock(); + for (int i = 0; i < m_clipList.count(); ++i) { + DocClipBase *clip = m_clipList.at(i); + if (clip->hasAudioThumb() && !clip->audioThumbCreated()) + m_audioThumbsQueue.append(m_clipList.at(i)->getId()); + } + m_thumbsMutex.unlock(); + if (!m_audioThumbsThread.isRunning() && !m_audioThumbsQueue.isEmpty()) { + m_audioThumbsThread = QtConcurrent::run(this, &ClipManager::slotGetAudioThumbs); } - if (m_generatingAudioId.isEmpty()) startAudioThumbsGeneration(); } void ClipManager::askForAudioThumb(const QString &id) { DocClipBase *clip = getClipById(id); - if (clip && KdenliveSettings::audiothumbnails()) { - m_audioThumbsQueue.append(id); - if (m_generatingAudioId.isEmpty()) startAudioThumbsGeneration(); + if (clip && KdenliveSettings::audiothumbnails() && (clip->hasAudioThumb())) { + m_thumbsMutex.lock(); + if (!m_audioThumbsQueue.contains(id)) m_audioThumbsQueue.append(id); + m_thumbsMutex.unlock(); + if (!m_audioThumbsThread.isRunning()) m_audioThumbsThread = QtConcurrent::run(this, &ClipManager::slotGetAudioThumbs); } } -void ClipManager::startAudioThumbsGeneration() +void ClipManager::slotGetAudioThumbs() { - if (!KdenliveSettings::audiothumbnails()) { - m_audioThumbsQueue.clear(); - m_generatingAudioId.clear(); - return; - } - if (!m_audioThumbsQueue.isEmpty()) { - m_generatingAudioId = m_audioThumbsQueue.takeFirst(); - DocClipBase *clip = getClipById(m_generatingAudioId); - if (!clip || !clip->slotGetAudioThumbs()) - endAudioThumbsGeneration(m_generatingAudioId); - } else { - m_generatingAudioId.clear(); - } -} + // We are in a new thread, so we need a new OpenGL context for the remainder of the function. + QGLWidget ctx(0, m_mainGLContext); + ctx.makeCurrent(); -void ClipManager::endAudioThumbsGeneration(const QString &requestedId) -{ - if (!KdenliveSettings::audiothumbnails()) { - m_audioThumbsQueue.clear(); - m_generatingAudioId.clear(); - return; - } - if (!m_audioThumbsQueue.isEmpty()) { - if (m_generatingAudioId == requestedId) { - startAudioThumbsGeneration(); + Mlt::Profile prof((char*) KdenliveSettings::current_profile().toUtf8().constData()); + mlt_audio_format audioFormat = mlt_audio_s16; + while (!m_abortAudioThumb && !m_audioThumbsQueue.isEmpty()) { + m_thumbsMutex.lock(); + m_processingAudioThumbId = m_audioThumbsQueue.takeFirst(); + m_thumbsMutex.unlock(); + DocClipBase *clip = getClipById(m_processingAudioThumbId); + if (!clip || clip->audioThumbCreated()) continue; + KUrl url = clip->fileURL(); + QString hash = clip->getClipHash(); + if (hash.isEmpty()) continue; + QString audioPath = projectFolder() + "/thumbs/" + hash + ".thumb"; + double lengthInFrames = clip->duration().frames(m_doc->fps()); + int frequency = 0; + int channels = 0; + QString data = clip->getProperty("frequency"); + if (!data.isEmpty()) frequency = data.toInt(); + if (frequency <= 0) frequency = 48000; + data = clip->getProperty("channels"); + if (!data.isEmpty()) channels = data.toInt(); + if (channels <= 0) channels = 2; + int arrayWidth = 20; + double frame = 0.0; + int maxVolume = 0; + audioByteArray storeIn; + QFile f(audioPath); + if (QFileInfo(audioPath).size() > 0 && f.open(QIODevice::ReadOnly)) { + const QByteArray channelarray = f.readAll(); + f.close(); + if (channelarray.size() != arrayWidth*(frame + lengthInFrames) * channels) { + kDebug() << "--- BROKEN THUMB FOR: " << url.fileName() << " ---------------------- "; + f.remove(); + continue; + } + kDebug() << "reading audio thumbs from file"; + + int h1 = arrayWidth * channels; + int h2 = (int) frame * h1; + int h3; + for (int z = (int) frame; z < (int)(frame + lengthInFrames) && !m_abortAudioThumb; z++) { + h3 = 0; + for (int c = 0; c < channels; c++) { + QByteArray audioArray(arrayWidth, '\x00'); + for (int i = 0; i < arrayWidth; ++i) { + audioArray[i] = channelarray.at(h2 + h3 + i); + if (audioArray.at(i) > maxVolume) maxVolume = audioArray.at(i); + } + h3 += arrayWidth; + storeIn[z][c] = audioArray; + } + h2 += h1; + } + if (!m_abortAudioThumb) { + clip->setProperty("audio_max", QString::number(maxVolume - 64)); + clip->updateAudioThumbnail(storeIn); + } + continue; + } + + if (!f.open(QIODevice::WriteOnly)) { + kDebug() << "++++++++ ERROR WRITING TO FILE: " << audioPath; + kDebug() << "++++++++ DISABLING AUDIO THUMBS"; + m_thumbsMutex.lock(); + m_audioThumbsQueue.clear(); + m_thumbsMutex.unlock(); + KdenliveSettings::setAudiothumbnails(false); + break; + } + + Mlt::Producer producer(prof, url.path().toUtf8().constData()); + if (!producer.is_valid()) { + kDebug() << "++++++++ INVALID CLIP: " << url.path(); + continue; + } + + producer.set("video_index", "-1"); + + if (KdenliveSettings::normaliseaudiothumbs()) { + /*Mlt::Filter m_convert(prof, "volume"); + m_convert.set("gain", "normalise"); + producer.attach(m_convert);*/ + } + + int last_val = 0; + double framesPerSecond = mlt_producer_get_fps(producer.get_producer()); + Mlt::Frame *mlt_frame; + + for (int z = (int) frame; z < (int)(frame + lengthInFrames) && producer.is_valid() && !m_abortAudioThumb; z++) { + int val = (int)((z - frame) / (frame + lengthInFrames) * 100.0); + if (last_val != val && val > 1) { + setThumbsProgress(i18n("Creating audio thumbnail for %1", url.fileName()), val); + last_val = val; + } + producer.seek(z); + mlt_frame = producer.get_frame(); + if (mlt_frame && mlt_frame->is_valid()) { + int samples = mlt_sample_calculator(framesPerSecond, frequency, mlt_frame->get_position()); + qint16* pcm = static_cast(mlt_frame->get_audio(audioFormat, frequency, channels, samples)); + for (int c = 0; c < channels; c++) { + QByteArray audioArray; + audioArray.resize(arrayWidth); + for (int i = 0; i < audioArray.size(); ++i) { + double pcmval = *(pcm + c + i * samples / audioArray.size()); + if (pcmval >= 0) { + pcmval = sqrt(pcmval) / 2.83 + 64; + audioArray[i] = pcmval; + if (pcmval > maxVolume) maxVolume = pcmval; + } + else { + pcmval = -sqrt(-pcmval) / 2.83 + 64; + audioArray[i] = pcmval; + if (-pcmval > maxVolume) maxVolume = -pcmval; + } + } + f.write(audioArray); + storeIn[z][c] = audioArray; + } + } else { + f.write(QByteArray(arrayWidth, '\x00')); + } + delete mlt_frame; + } + f.close(); + setThumbsProgress(i18n("Creating audio thumbnail for %1", url.fileName()), -1); + if (m_abortAudioThumb) { + f.remove(); + } else { + clip->updateAudioThumbnail(storeIn); + clip->setProperty("audio_max", QString::number(maxVolume - 64)); } - } else { - m_generatingAudioId.clear(); } + m_processingAudioThumbId.clear(); } void ClipManager::setThumbsProgress(const QString &message, int progress) @@ -233,7 +413,7 @@ QMap ClipManager::documentFolderList() const void ClipManager::addClip(DocClipBase *clip) { m_clipList.append(clip); - if (clip->clipType() != COLOR && clip->clipType() != SLIDESHOW && !clip->fileURL().isEmpty()) { + if (clip->clipType() != Color && clip->clipType() != SlideShow && !clip->fileURL().isEmpty()) { // listen for file change //kDebug() << "// LISTEN FOR: " << clip->fileURL().path(); m_fileWatcher.addFile(clip->fileURL().path()); @@ -249,7 +429,7 @@ void ClipManager::slotDeleteClips(QStringList ids) QUndoCommand *delClips = new QUndoCommand(); delClips->setText(i18np("Delete clip", "Delete clips", ids.size())); - for (int i = 0; i < ids.size(); i++) { + for (int i = 0; i < ids.size(); ++i) { DocClipBase *clip = getClipById(ids.at(i)); if (clip) { new AddClipCommand(m_doc, clip->toXML(), ids.at(i), false, delClips); @@ -260,14 +440,14 @@ void ClipManager::slotDeleteClips(QStringList ids) void ClipManager::deleteClip(const QString &clipId) { - for (int i = 0; i < m_clipList.count(); i++) { + for (int i = 0; i < m_clipList.count(); ++i) { if (m_clipList.at(i)->getId() == clipId) { - if (m_clipList.at(i)->clipType() != COLOR && m_clipList.at(i)->clipType() != SLIDESHOW && !m_clipList.at(i)->fileURL().isEmpty()) { + DocClipBase *clip = m_clipList.takeAt(i); + if (clip->clipType() != Color && clip->clipType() != SlideShow && !clip->fileURL().isEmpty()) { //if (m_clipList.at(i)->clipType() == IMAGE || m_clipList.at(i)->clipType() == AUDIO || (m_clipList.at(i)->clipType() == TEXT && !m_clipList.at(i)->fileURL().isEmpty())) { // listen for file change - m_fileWatcher.removeFile(m_clipList.at(i)->fileURL().path()); + m_fileWatcher.removeFile(clip->fileURL().path()); } - DocClipBase *clip = m_clipList.takeAt(i); delete clip; clip = NULL; break; @@ -284,7 +464,7 @@ DocClipBase *ClipManager::getClipById(QString clipId) { //kDebug() << "++++ CLIP MAN, LOOKING FOR CLIP ID: " << clipId; clipId = clipId.section('_', 0, 0); - for (int i = 0; i < m_clipList.count(); i++) { + for (int i = 0; i < m_clipList.count(); ++i) { if (m_clipList.at(i)->getId() == clipId) { //kDebug() << "++++ CLIP MAN, FOUND FOR CLIP ID: " << clipId; return m_clipList.at(i); @@ -293,12 +473,12 @@ DocClipBase *ClipManager::getClipById(QString clipId) return NULL; } -const QList ClipManager::getClipByResource(QString resource) +const QList ClipManager::getClipByResource(const QString &resource) { QList list; QString clipResource; QString proxyResource; - for (int i = 0; i < m_clipList.count(); i++) { + for (int i = 0; i < m_clipList.count(); ++i) { clipResource = m_clipList.at(i)->getProperty("resource"); proxyResource = m_clipList.at(i)->getProperty("proxy"); if (clipResource.isEmpty()) clipResource = m_clipList.at(i)->getProperty("colour"); @@ -312,14 +492,14 @@ const QList ClipManager::getClipByResource(QString resource) void ClipManager::clearUnusedProducers() { - for (int i = 0; i < m_clipList.count(); i++) { + for (int i = 0; i < m_clipList.count(); ++i) { if (m_clipList.at(i)->numReferences() == 0) m_clipList.at(i)->deleteProducers(); } } void ClipManager::resetProducersList(const QList prods, bool displayRatioChanged, bool fpsChanged) { - for (int i = 0; i < m_clipList.count(); i++) { + for (int i = 0; i < m_clipList.count(); ++i) { if (m_clipList.at(i)->numReferences() > 0 || displayRatioChanged || fpsChanged) { m_clipList.at(i)->deleteProducers(); } @@ -327,7 +507,7 @@ void ClipManager::resetProducersList(const QList prods, bool d QString id; Mlt::Producer *prod; QStringList brokenClips; - for (int i = 0; i < prods.count(); i++) { + for (int i = 0; i < prods.count(); ++i) { prod = prods.at(i); id = prod->get("id"); if (id.contains('_')) id = id.section('_', 0, 0); @@ -345,16 +525,47 @@ void ClipManager::resetProducersList(const QList prods, bool d emit checkAllClips(displayRatioChanged, fpsChanged, brokenClips); } -void ClipManager::slotAddClipList(const KUrl::List urls, const QString &group, const QString &groupId) +void ClipManager::slotAddClip(KIO::Job *job, const KUrl &, const KUrl &dst) { - QUndoCommand *addClips = new QUndoCommand(); + KIO::MetaData meta = job->metaData(); + QMap data; + data.insert("group", meta.value("group")); + data.insert("groupid", meta.value("groupid")); + data.insert("comment", meta.value("comment")); + kDebug()<<"Finished copying: "< &data) +{ + QUndoCommand *addClips = new QUndoCommand(); + // Update list of removable volumes + //TODO: update only when new volume is plugged / unplugged + listRemovableVolumes(); foreach(const KUrl & file, urls) { - if (KIO::NetAccess::exists(file, KIO::NetAccess::SourceSide, NULL)) { - if (!getClipByResource(file.path()).empty()) { + if (QFile::exists(file.path())) {//KIO::NetAccess::exists(file, KIO::NetAccess::SourceSide, NULL)) { + if (!data.contains("bypassDuplicate") && !getClipByResource(file.path()).empty()) { if (KMessageBox::warningContinueCancel(kapp->activeWindow(), i18n("Clip %1
already exists in project, what do you want to do?", file.path()), i18n("Clip already exists")) == KMessageBox::Cancel) continue; } + if (isOnRemovableDevice(file)) { + int answer = KMessageBox::warningYesNoCancel(kapp->activeWindow(), i18n("Clip %1
is on a removable device, will not be available when device is unplugged", file.path()), i18n("File on a Removable Device"), KGuiItem(i18n("Copy file to project folder")), KGuiItem(i18n("Continue")), KStandardGuiItem::cancel(), QString("copyFilesToProjectFolder")); + if (answer == KMessageBox::Cancel) continue; + else if (answer == KMessageBox::Yes) { + // Copy files to project folder + QString sourcesFolder = m_doc->projectFolder().path(KUrl::AddTrailingSlash) + "clips/"; + KIO::NetAccess::mkdir(sourcesFolder, kapp->activeWindow()); + //KIO::filesize_t m_requestedSize; + KIO::CopyJob *copyjob = KIO::copy (file, KUrl(sourcesFolder)); + //TODO: for some reason, passing metadata does not work... + copyjob->addMetaData("group", data.value("group")); + copyjob->addMetaData("groupId", data.value("groupId")); + copyjob->addMetaData("comment", data.value("comment")); + copyjob->ui()->setWindow(kapp->activeWindow()); + connect(copyjob, SIGNAL(copyingDone(KIO::Job*,KUrl,KUrl,time_t,bool,bool)), this, SLOT(slotAddClip(KIO::Job*,KUrl,KUrl))); + continue; + } + } kDebug() << "Adding clip: " << file.path(); QDomDocument doc; QDomElement prod = doc.createElement("producer"); @@ -362,22 +573,31 @@ void ClipManager::slotAddClipList(const KUrl::List urls, const QString &group, c prod.setAttribute("resource", file.path()); uint id = m_clipIdCounter++; prod.setAttribute("id", QString::number(id)); - if (!group.isEmpty()) { - prod.setAttribute("groupname", group); - prod.setAttribute("groupid", groupId); + if (data.contains("comment")) prod.setAttribute("description", data.value("comment")); + if (data.contains("group")) { + prod.setAttribute("groupname", data.value("group")); + prod.setAttribute("groupid", data.value("groupId")); } + if (data.contains("video_index")) prod.setAttribute("video_index", data.value("video_index")); + if (data.contains("audio_index")) prod.setAttribute("audio_index", data.value("audio_index")); + KMimeType::Ptr type = KMimeType::findByUrl(file); if (type->name().startsWith("image/")) { - prod.setAttribute("type", (int) IMAGE); + prod.setAttribute("type", (int) Image); prod.setAttribute("in", 0); - prod.setAttribute("out", m_doc->getFramePos(KdenliveSettings::image_duration())); + prod.setAttribute("out", m_doc->getFramePos(KdenliveSettings::image_duration()) - 1); if (KdenliveSettings::autoimagetransparency()) prod.setAttribute("transparency", 1); // Read EXIF metadata for JPEG if (type->is("image/jpeg")) { KFileMetaInfo metaInfo(file.path(), QString("image/jpeg"), KFileMetaInfo::TechnicalInfo); const QHash metaInfoItems = metaInfo.items(); foreach(const KFileMetaInfoItem & metaInfoItem, metaInfoItems) { - prod.setAttribute("meta.attr." + metaInfoItem.name().section("#", 1), metaInfoItem.value().toString()); + QDomElement meta = doc.createElement("metaproperty"); + meta.setAttribute("name", "meta.attr." + metaInfoItem.name().section('#', 1)); + QDomText value = doc.createTextNode(metaInfoItem.value().toString()); + meta.setAttribute("tool", "KDE Metadata"); + meta.appendChild(value); + prod.appendChild(meta); } } } else if (type->is("application/x-kdenlivetitle")) { @@ -386,10 +606,10 @@ void ClipManager::slotAddClipList(const KUrl::List urls, const QString &group, c QFile txtfile(file.path()); if (txtfile.open(QIODevice::ReadOnly) && txtdoc.setContent(&txtfile)) { txtfile.close(); - prod.setAttribute("type", (int) TEXT); + prod.setAttribute("type", (int) Text); // extract embeded images QDomNodeList items = txtdoc.elementsByTagName("content"); - for (int i = 0; i < items.count() ; i++) { + for (int i = 0; i < items.count() ; ++i) { QDomElement content = items.item(i).toElement(); if (content.hasAttribute("base64")) { QString titlesFolder = m_doc->projectFolder().path(KUrl::AddTrailingSlash) + "titles/"; @@ -400,20 +620,29 @@ void ClipManager::slotAddClipList(const KUrl::List urls, const QString &group, c } } } - QString titleData = txtdoc.toString(); - prod.setAttribute("xmldata", titleData); prod.setAttribute("transparency", 1); prod.setAttribute("in", 0); - int out = txtdoc.documentElement().attribute("out").toInt(); - if (out > 0) - prod.setAttribute("out", out); - else - prod.setAttribute("out", m_doc->getFramePos(KdenliveSettings::title_duration())); + if (!txtdoc.documentElement().hasAttribute("out")) { + prod.setAttribute("out", m_doc->getFramePos(KdenliveSettings::title_duration()) - 1); + txtdoc.documentElement().setAttribute("out", m_doc->getFramePos(KdenliveSettings::title_duration()) - 1); + } + else { + int out = txtdoc.documentElement().attribute("out").toInt(); + if (out >= 0) + prod.setAttribute("out", out); + else { + prod.setAttribute("out", m_doc->getFramePos(KdenliveSettings::title_duration()) - 1); + txtdoc.documentElement().setAttribute("out", m_doc->getFramePos(KdenliveSettings::title_duration()) - 1); + } + } + QString titleData = txtdoc.toString(); + prod.setAttribute("xmldata", titleData); } else txtfile.close(); } new AddClipCommand(m_doc, doc.documentElement(), QString::number(id), true, addClips); } + else kDebug()<<"// CANNOT READ FILE: "<childCount() > 0) { addClips->setText(i18np("Add clip", "Add clips", addClips->childCount())); @@ -421,9 +650,9 @@ void ClipManager::slotAddClipList(const KUrl::List urls, const QString &group, c } } -void ClipManager::slotAddClipFile(const KUrl &url, const QString &group, const QString &groupId) +void ClipManager::slotAddClipFile(const KUrl &url, const QMap &data) { - slotAddClipList(KUrl::List(url), group, groupId); + slotAddClipList(KUrl::List(url), data); } void ClipManager::slotAddXmlClipFile(const QString &name, const QDomElement &xml, const QString &group, const QString &groupId) @@ -431,7 +660,7 @@ void ClipManager::slotAddXmlClipFile(const QString &name, const QDomElement &xml QDomDocument doc; doc.appendChild(doc.importNode(xml, true)); QDomElement prod = doc.documentElement(); - prod.setAttribute("type", (int) PLAYLIST); + prod.setAttribute("type", (int) Playlist); uint id = m_clipIdCounter++; prod.setAttribute("id", QString::number(id)); prod.setAttribute("name", name); @@ -443,18 +672,18 @@ void ClipManager::slotAddXmlClipFile(const QString &name, const QDomElement &xml m_doc->commandStack()->push(command); } -void ClipManager::slotAddColorClipFile(const QString &name, const QString &color, QString duration, const QString &group, const QString &groupId) +void ClipManager::slotAddColorClipFile(const QString &name, const QString &color, const QString &duration, const QString &group, const QString &groupId) { QDomDocument doc; QDomElement prod = doc.createElement("producer"); doc.appendChild(prod); prod.setAttribute("mlt_service", "colour"); prod.setAttribute("colour", color); - prod.setAttribute("type", (int) COLOR); + prod.setAttribute("type", (int) Color); uint id = m_clipIdCounter++; prod.setAttribute("id", QString::number(id)); prod.setAttribute("in", "0"); - prod.setAttribute("out", m_doc->getFramePos(duration)); + prod.setAttribute("out", m_doc->getFramePos(duration) - 1); prod.setAttribute("name", name); if (!group.isEmpty()) { prod.setAttribute("groupname", group); @@ -464,29 +693,18 @@ void ClipManager::slotAddColorClipFile(const QString &name, const QString &color m_doc->commandStack()->push(command); } -void ClipManager::slotAddSlideshowClipFile(const QString &name, const QString &path, int count, const QString &duration, - const bool loop, const bool crop, const bool fade, - const QString &luma_duration, const QString &luma_file, const int softness, - const QString &animation, const QString &group, const QString &groupId) +void ClipManager::slotAddSlideshowClipFile(QMap properties, const QString &group, const QString &groupId) { QDomDocument doc; QDomElement prod = doc.createElement("producer"); doc.appendChild(prod); - prod.setAttribute("resource", path); - prod.setAttribute("type", (int) SLIDESHOW); + QMap::const_iterator i = properties.constBegin(); + while (i != properties.constEnd()) { + prod.setAttribute(i.key(), i.value()); + ++i; + } + prod.setAttribute("type", (int) SlideShow); uint id = m_clipIdCounter++; - prod.setAttribute("id", QString::number(id)); - prod.setAttribute("in", "0"); - prod.setAttribute("out", m_doc->getFramePos(duration) * count); - prod.setAttribute("ttl", m_doc->getFramePos(duration)); - prod.setAttribute("luma_duration", m_doc->getFramePos(luma_duration)); - prod.setAttribute("name", name); - prod.setAttribute("loop", loop); - prod.setAttribute("crop", crop); - prod.setAttribute("fade", fade); - prod.setAttribute("softness", QString::number(softness)); - prod.setAttribute("luma_file", luma_file); - prod.setAttribute("animation", animation); if (!group.isEmpty()) { prod.setAttribute("groupname", group); prod.setAttribute("groupid", groupId); @@ -497,7 +715,7 @@ void ClipManager::slotAddSlideshowClipFile(const QString &name, const QString &p -void ClipManager::slotAddTextClipFile(const QString &titleName, int out, const QString &xml, const QString &group, const QString &groupId) +void ClipManager::slotAddTextClipFile(const QString &titleName, int duration, const QString &xml, const QString &group, const QString &groupId) { QDomDocument doc; QDomElement prod = doc.createElement("producer"); @@ -511,10 +729,10 @@ void ClipManager::slotAddTextClipFile(const QString &titleName, int out, const Q prod.setAttribute("groupname", group); prod.setAttribute("groupid", groupId); } - prod.setAttribute("type", (int) TEXT); + prod.setAttribute("type", (int) Text); prod.setAttribute("transparency", "1"); prod.setAttribute("in", "0"); - prod.setAttribute("out", out); + prod.setAttribute("out", duration - 1); AddClipCommand *command = new AddClipCommand(m_doc, doc.documentElement(), QString::number(id), true); m_doc->commandStack()->push(command); } @@ -532,20 +750,26 @@ void ClipManager::slotAddTextTemplateClip(QString titleName, const KUrl &path, c prod.setAttribute("groupname", group); prod.setAttribute("groupid", groupId); } - prod.setAttribute("type", (int) TEXT); + prod.setAttribute("type", (int) Text); prod.setAttribute("transparency", "1"); prod.setAttribute("in", "0"); - int out = 0; + int duration = 0; QDomDocument titledoc; QFile txtfile(path.path()); if (txtfile.open(QIODevice::ReadOnly) && titledoc.setContent(&txtfile)) { txtfile.close(); - out = titledoc.documentElement().attribute("out").toInt(); + if (titledoc.documentElement().hasAttribute("duration")) { + duration = titledoc.documentElement().attribute("duration").toInt(); + } + else { + // keep some time for backwards compatibility - 26/12/12 + duration = titledoc.documentElement().attribute("out").toInt(); + } } else txtfile.close(); - if (out == 0) out = m_doc->getFramePos(KdenliveSettings::image_duration()); - prod.setAttribute("out", out); + if (duration == 0) duration = m_doc->getFramePos(KdenliveSettings::title_duration()); + prod.setAttribute("out", duration - 1); AddClipCommand *command = new AddClipCommand(m_doc, doc.documentElement(), QString::number(id), true); m_doc->commandStack()->push(command); @@ -598,20 +822,20 @@ QDomElement ClipManager::groupsXml() const QDomDocument doc; QDomElement groups = doc.createElement("groups"); doc.appendChild(groups); - for (int i = 0; i < m_groupsList.count(); i++) { + for (int i = 0; i < m_groupsList.count(); ++i) { QDomElement group = doc.createElement("group"); groups.appendChild(group); QList children = m_groupsList.at(i)->childItems(); for (int j = 0; j < children.count(); j++) { - if (children.at(j)->type() == AVWIDGET || children.at(j)->type() == TRANSITIONWIDGET) { + if (children.at(j)->type() == AVWidget || children.at(j)->type() == TransitionWidget) { AbstractClipItem *item = static_cast (children.at(j)); ItemInfo info = item->info(); - if (item->type() == AVWIDGET) { + if (item->type() == AVWidget) { QDomElement clip = doc.createElement("clipitem"); clip.setAttribute("track", info.track); clip.setAttribute("position", info.startPos.frames(m_doc->fps())); group.appendChild(clip); - } else if (item->type() == TRANSITIONWIDGET) { + } else if (item->type() == TransitionWidget) { QDomElement clip = doc.createElement("transitionitem"); clip.setAttribute("track", info.track); clip.setAttribute("position", info.startPos.frames(m_doc->fps())); @@ -628,7 +852,7 @@ void ClipManager::slotClipModified(const QString &path) { //kDebug() << "// CLIP: " << path << " WAS MODIFIED"; const QList list = getClipByResource(path); - for (int i = 0; i < list.count(); i++) { + for (int i = 0; i < list.count(); ++i) { DocClipBase *clip = list.at(i); if (clip != NULL) { QString id = clip->getId(); @@ -660,7 +884,7 @@ void ClipManager::slotClipMissing(const QString &path) { // kDebug() << "// CLIP: " << path << " WAS MISSING"; const QList list = getClipByResource(path); - for (int i = 0; i < list.count(); i++) { + for (int i = 0; i < list.count(); ++i) { DocClipBase *clip = list.at(i); if (clip != NULL) emit missingClip(clip->getId()); } @@ -670,7 +894,7 @@ void ClipManager::slotClipAvailable(const QString &path) { // kDebug() << "// CLIP: " << path << " WAS ADDED"; const QList list = getClipByResource(path); - for (int i = 0; i < list.count(); i++) { + for (int i = 0; i < list.count(); ++i) { DocClipBase *clip = list.at(i); if (clip != NULL) emit availableClip(clip->getId()); } @@ -682,3 +906,101 @@ int ClipManager::clipsCount() const } +void ClipManager::listRemovableVolumes() +{ + QList volumes; + m_removableVolumes.clear(); + + QList devices = Solid::Device::listFromType(Solid::DeviceInterface::StorageAccess); + + foreach(const Solid::Device &accessDevice, devices) + { + // check for StorageAccess + if (!accessDevice.is()) + continue; + + const Solid::StorageAccess *access = accessDevice.as(); + + if (!access->isAccessible()) + continue; + + // check for StorageDrive + Solid::Device driveDevice; + for (Solid::Device currentDevice = accessDevice; currentDevice.isValid(); currentDevice = currentDevice.parent()) + { + if (currentDevice.is()) + { + driveDevice = currentDevice; + break; + } + } + if (!driveDevice.isValid()) + continue; + + Solid::StorageDrive *drive = driveDevice.as(); + if (!drive->isRemovable()) continue; + + // check for StorageVolume + Solid::Device volumeDevice; + for (Solid::Device currentDevice = accessDevice; currentDevice.isValid(); currentDevice = currentDevice.parent()) + { + if (currentDevice.is()) + { + volumeDevice = currentDevice; + break; + } + } + if (!volumeDevice.isValid()) + continue; + + Solid::StorageVolume *volume = volumeDevice.as(); + + SolidVolumeInfo info; + info.path = access->filePath(); + info.isMounted = access->isAccessible(); + if (!info.path.isEmpty() && !info.path.endsWith('/')) + info.path += '/'; + info.uuid = volume->uuid(); + info.label = volume->label(); + info.isRemovable = drive->isRemovable(); + m_removableVolumes << info; + } +} + +bool ClipManager::isOnRemovableDevice(const KUrl &url) +{ + //SolidVolumeInfo volume; + QString path = url.path(KUrl::RemoveTrailingSlash); + int volumeMatch = 0; + + //FIXME: Network shares! Here we get only the volume of the mount path... + // This is probably not really clean. But Solid does not help us. + foreach (const SolidVolumeInfo &v, m_removableVolumes) + { + if (v.isMounted && !v.path.isEmpty() && path.startsWith(v.path)) + { + int length = v.path.length(); + if (length > volumeMatch) + { + volumeMatch = v.path.length(); + //volume = v; + } + } + } + + return volumeMatch; +} + +void ClipManager::projectTreeThumbReady(const QString &id, int frame, const QImage &img, int type) +{ + switch (type) { + case 2: + emit gotClipPropertyThumbnail(id, img); + break; + default: + emit thumbReady(id, frame, img); + } +} + + +#include "clipmanager.moc"