X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Fclipstabilize.cpp;h=c35586a8556031717a4f90567b4ea5577fee6638;hb=c3302003093710ee247ad84c0fe2ef3c579d417f;hp=4ec8c27c69880c082c14f89261c8ec2fde821b7a;hpb=7f71c1eb23f4d2a9c32696a2a4394ce672b02e1b;p=kdenlive diff --git a/src/clipstabilize.cpp b/src/clipstabilize.cpp index 4ec8c27c..c35586a8 100644 --- a/src/clipstabilize.cpp +++ b/src/clipstabilize.cpp @@ -20,65 +20,115 @@ #include "clipstabilize.h" +#include "doubleparameterwidget.h" #include #include #include "kdenlivesettings.h" #include #include +#include #include #include +#include #include - -ClipStabilize::ClipStabilize(KUrl::List urls, const QString ¶ms, QWidget * parent) : - QDialog(parent), m_urls(urls), m_duration(0),m_profile(NULL),m_consumer(NULL),m_playlist(NULL) +ClipStabilize::ClipStabilize(const QStringList &urls, const QString &filterName,QWidget * parent) : + QDialog(parent), + m_filtername(filterName), + m_urls(urls), + vbox(NULL) { setFont(KGlobalSettings::toolBarFont()); setupUi(this); - setAttribute(Qt::WA_DeleteOnClose); - log_text->setHidden(true); setWindowTitle(i18n("Stabilize Clip")); - auto_add->setText(i18np("Add clip to project", "Add clips to project", m_urls.count())); - m_profile = new Mlt::Profile(KdenliveSettings::current_profile().toUtf8().constData()); - filtername=params; + auto_add->setText(i18np("Add clip to project", "Add clips to project", urls.count())); + auto_add->setChecked(KdenliveSettings::add_new_clip()); + + QPalette p = palette(); + KColorScheme scheme(p.currentColorGroup(), KColorScheme::View, KSharedConfig::openConfig(KdenliveSettings::colortheme())); + QColor dark_bg = scheme.shade(KColorScheme::DarkShade); + QColor selected_bg = scheme.decoration(KColorScheme::FocusColor).color(); + QColor hover_bg = scheme.decoration(KColorScheme::HoverColor).color(); + QColor light_bg = scheme.shade(KColorScheme::LightShade); + + QString stylesheet(QString("QProgressBar:horizontal {border: 1px solid %1;border-radius:0px;border-top-left-radius: 4px;border-bottom-left-radius: 4px;border-right: 0px;background:%4;padding: 0px;text-align:left center}\ + QProgressBar:horizontal#dragOnly {background: %1} QProgressBar:horizontal:hover#dragOnly {background: %3} QProgressBar:horizontal:hover {border: 1px solid %3;border-right: 0px;}\ + QProgressBar::chunk:horizontal {background: %1;} QProgressBar::chunk:horizontal:hover {background: %3;}\ + QProgressBar:horizontal[inTimeline=\"true\"] { border: 1px solid %2;border-right: 0px;background: %4;padding: 0px;text-align:left center } QProgressBar::chunk:horizontal[inTimeline=\"true\"] {background: %2;}\ + QAbstractSpinBox#dragBox {border: 1px solid %1;border-top-right-radius: 4px;border-bottom-right-radius: 4px;padding-right:0px;} QAbstractSpinBox::down-button#dragBox {width:0px;padding:0px;}\ + QAbstractSpinBox::up-button#dragBox {width:0px;padding:0px;} QAbstractSpinBox[inTimeline=\"true\"]#dragBox { border: 1px solid %2;} QAbstractSpinBox:hover#dragBox {border: 1px solid %3;} ") + .arg(dark_bg.name()).arg(selected_bg.name()).arg(hover_bg.name()).arg(light_bg.name())); + setStyleSheet(stylesheet); + + if (m_urls.count() == 1) { + QString newFile = m_urls.first(); + newFile.append(".mlt"); + KUrl dest(newFile); + dest_url->setMode(KFile::File); + dest_url->setUrl(KUrl(newFile)); + dest_url->fileDialog()->setOperationMode(KFileDialog::Saving); + +} else { + label_dest->setText(i18n("Destination folder")); + dest_url->setMode(KFile::Directory); + dest_url->setUrl(KUrl(KUrl(m_urls.first()).directory())); + dest_url->fileDialog()->setOperationMode(KFileDialog::Saving); +} + + if (m_filtername=="videostab"){ + QStringList ls; + ls << "shutterangle,type,int,value,0,min,0,max,180,tooltip,Angle that Images could be maximum rotated"; + fillParameters(ls); +}else if (m_filtername=="videostab2"){ + // Some default params have to be set: + m_fixedParams << "algo=1" << "relative=1"; + QStringList ls; + ls << "accuracy,type,int,value,8,min,1,max,10,tooltip,Accuracy of Shakiness detection"; + ls << "shakiness,type,int,value,4,min,1,max,10,tooltip,How shaky is the Video"; + ls << "stepsize,type,int,value,6,min,0,max,100,tooltip,Stepsize of Detection process minimum around"; + //ls << "algo,type,bool,value,1,min,0,max,1,tooltip,0 = Bruteforce 1 = small measurement fields"; + ls << "mincontrast,type,double,value,0.3,min,0,max,1,factor,1,decimals,2,tooltip,Below this Contrast Field is discarded"; + //ls << "show,type,int,value,0,min,0,max,2,tooltip,0 = draw nothing. 1 or 2 show fields and transforms"; + ls << "smoothing,type,int,value,10,min,0,max,100,tooltip,number of frames for lowpass filtering"; + ls << "maxshift,type,int,value,-1,min,-1,max,1000,tooltip,max number of pixels to shift"; + ls << "maxangle,type,double,value,-1,min,-1,max,3.14,decimals,2,tooltip,max angle to rotate (in rad)"; + ls << "crop,type,bool,value,0,min,0,max,1,tooltip,0 = keep border 1 = black background"; + //ls << "invert,type,bool,value,0,min,0,max,1,tooltip,invert transform"; + //ls << "relative,type,bool,value,1,min,0,max,1,tooltip,0 = absolute transform 1= relative"; + ls << "zoom,type,int,value,0,min,-500,max,500,tooltip,additional zoom during transform"; + ls << "optzoom,type,bool,value,1,min,0,max,1,tooltip,use optimal zoom (calulated from transforms)"; + ls << "sharpen,type,double,value,0.8,min,0,max,1,decimals,1,tooltip,sharpen transformed image"; + fillParameters(ls); - if (m_urls.count() == 1) { - QString fileName = m_urls.at(0).path(); //.section('.', 0, -1); - QString newFile = fileName.append(".mlt"); - KUrl dest(newFile); - source_url->setUrl(m_urls.at(0)); - dest_url->setMode(KFile::File); - dest_url->setUrl(dest); - dest_url->fileDialog()->setOperationMode(KFileDialog::Saving); - urls_list->setHidden(true); - connect(source_url, SIGNAL(textChanged(const QString &)), this, SLOT(slotUpdateParams())); - } else { - label_source->setHidden(true); - source_url->setHidden(true); - label_dest->setText(i18n("Destination folder")); - dest_url->setMode(KFile::Directory); - dest_url->setUrl(KUrl(m_urls.at(0).directory())); - dest_url->fileDialog()->setOperationMode(KFileDialog::Saving); - for (int i = 0; i < m_urls.count(); i++) - urls_list->addItem(m_urls.at(i).path()); - } - if (!params.isEmpty()) { - label_profile->setHidden(true); - profile_list->setHidden(true); - //ffmpeg_params->setPlainText(params.simplified()); - /*if (!description.isEmpty()) { - transcode_info->setText(description); - } else transcode_info->setHidden(true);*/ - } - - connect(button_start, SIGNAL(clicked()), this, SLOT(slotStartStabilize())); - connect(buttonBox,SIGNAL(rejected()), this, SLOT(slotAbortStabilize())); - - m_timer=new QTimer(this); - connect(m_timer, SIGNAL(timeout()), this, SLOT(slotShowStabilizeInfo())); - - adjustSize(); +} + + connect(buttonBox->button(QDialogButtonBox::Ok),SIGNAL(clicked()), this, SLOT(slotValidate())); + + vbox=new QVBoxLayout(optionsbox); + QHashIterator > hi(m_ui_params); + while(hi.hasNext()){ + hi.next(); + QHash val=hi.value(); + if (val["type"]=="int" || val["type"]=="double"){ + DoubleParameterWidget *dbl=new DoubleParameterWidget(hi.key(), val["value"].toDouble(), + val["min"].toDouble(),val["max"].toDouble(),val["value"].toDouble(), + "",0/*id*/,""/*suffix*/,val["decimals"]!=""?val["decimals"].toInt():0,this); + dbl->setObjectName(hi.key()); + dbl->setToolTip(val["tooltip"]); + connect(dbl,SIGNAL(valueChanged(double)),this,SLOT(slotUpdateParams())); + vbox->addWidget(dbl); + }else if (val["type"]=="bool"){ + QCheckBox *ch=new QCheckBox(hi.key(),this); + ch->setCheckState(val["value"] == "0" ? Qt::Unchecked : Qt::Checked); + ch->setObjectName(hi.key()); + connect(ch, SIGNAL(stateChanged(int)) , this,SLOT(slotUpdateParams())); + ch->setToolTip(val["tooltip"]); + vbox->addWidget(ch); + + } +} +adjustSize(); } ClipStabilize::~ClipStabilize() @@ -86,23 +136,60 @@ ClipStabilize::~ClipStabilize() /*if (m_stabilizeProcess.state() != QProcess::NotRunning) { m_stabilizeProcess.close(); }*/ - if (m_profile) free (m_profile); - if (m_consumer) free (m_consumer); - if (m_playlist) free (m_playlist); + KdenliveSettings::setAdd_new_clip(auto_add->isChecked()); +} + +QStringList ClipStabilize::params() +{ + //we must return a stringlist with: + // producerparams << filtername << filterparams << consumer << consumerparams + QStringList params; + // producer params + params << QString(); + // filter + params << m_filtername; + QStringList filterparamsList = m_fixedParams; + QHashIterator > it(m_ui_params); + while (it.hasNext()){ + it.next(); + filterparamsList << it.key() + '=' + it.value().value("value"); + } + params << filterparamsList.join(" "); + + // consumer + params << "xml"; + // consumer params + QString title = i18n("Stabilised"); + params << QString("all=1 title=\"%1\"").arg(title); + return params; +} + +QString ClipStabilize::destination() const +{ + if (m_urls.count() == 1) + return dest_url->url().path(); + else + return dest_url->url().path(KUrl::AddTrailingSlash); +} + +QString ClipStabilize::desc() const +{ + return i18n("Stabilize clip"); } void ClipStabilize::slotStartStabilize() { + /* if (m_consumer && !m_consumer->is_stopped()) { return; } m_duration = 0; QStringList parameters; QString destination; - QString params = ffmpeg_params->toPlainText().simplified(); + //QString params = ffmpeg_params->toPlainText().simplified(); if (urls_list->count() > 0) { source_url->setUrl(m_urls.takeFirst()); - destination = dest_url->url().path(); + destination = dest_url->url().path(KUrl::AddTrailingSlash)+ source_url->url().fileName()+".mlt"; QList matching = urls_list->findItems(source_url->url().path(), Qt::MatchExactly); if (matching.count() > 0) { matching.at(0)->setFlags(Qt::ItemIsSelectable); @@ -114,83 +201,101 @@ void ClipStabilize::slotStartStabilize() QString s_url = source_url->url().path(); if (QFile::exists(destination)) { - if (KMessageBox::questionYesNo(this, i18n("File %1 already exists.\nDo you want to overwrite it?", destination )) == KMessageBox::No) return; + if (KMessageBox::questionYesNo(this, i18n("File %1 already exists.\nDo you want to overwrite it?", destination )) == KMessageBox::No) return; } buttonBox->button(QDialogButtonBox::Abort)->setText(i18n("Abort")); - if (m_profile){ - qDebug() << m_profile->description(); - m_playlist= new Mlt::Playlist; - Mlt::Filter filter(*m_profile,filtername.toUtf8().data()); - Mlt::Producer p(*m_profile,s_url.toUtf8().data()); - m_playlist->append(p); - m_playlist->attach(filter); - //m_producer->set("out",20); - m_consumer= new Mlt::Consumer(*m_profile,"xml",destination.toUtf8().data()); - m_consumer->set("all","1"); - m_consumer->set("real_time","-2"); - m_consumer->connect(*m_playlist); - QtConcurrent::run(this, &ClipStabilize::slotRunStabilize); - button_start->setEnabled(false); - } - + if (m_profile){ + m_playlist= new Mlt::Playlist; + Mlt::Filter filter(*m_profile,filtername.toUtf8().data()); + QHashIterator > it(m_ui_params); + while (it.hasNext()){ + it.next(); + filter.set( + it.key().toAscii().data(), + QString::number(it.value()["value"].toDouble()).toAscii().data()); + } + Mlt::Producer p(*m_profile,s_url.toUtf8().data()); + if (p.is_valid()) { + m_playlist->append(p); + m_playlist->attach(filter); + m_consumer= new Mlt::Consumer(*m_profile,"xml",destination.toUtf8().data()); + m_consumer->set("all",1); + m_consumer->set("real_time",-2); + m_consumer->connect(*m_playlist); + m_stabilizeRun = QtConcurrent::run(this, &ClipStabilize::slotRunStabilize); + m_timer->start(500); + button_start->setEnabled(false); + } + } +*/ } -void ClipStabilize::slotRunStabilize() -{ - if (m_consumer) - { - m_timer->start(500); - m_consumer->run(); - } -} -void ClipStabilize::slotAbortStabilize() + +void ClipStabilize::slotUpdateParams() { - if (m_consumer) - { - m_timer->stop(); - m_consumer->stop(); - slotStabilizeFinished(false); - } + for (int i=0;icount();++i){ + QWidget* w=vbox->itemAt(i)->widget(); + QString name=w->objectName(); + if (!name.isEmpty() && m_ui_params.contains(name)){ + if (m_ui_params[name]["type"]=="int" || m_ui_params[name]["type"]=="double"){ + DoubleParameterWidget *dbl=(DoubleParameterWidget*)w; + m_ui_params[name]["value"]=QString::number((double)(dbl->getValue())); + }else if (m_ui_params[name]["type"]=="bool"){ + QCheckBox *ch=(QCheckBox*)w; + m_ui_params[name]["value"]= ch->checkState() == Qt::Checked ? "1" : "0" ; + } + } + } } -void ClipStabilize::slotShowStabilizeInfo() +bool ClipStabilize::autoAddClip() const { - if (m_playlist){ - job_progress->setValue((int) (100.0 * m_consumer->position()/m_playlist->get_out() )); - if (m_consumer->position()==m_playlist->get_out()){ - m_timer->stop(); - slotStabilizeFinished(true); - } - } + return auto_add->isChecked(); } -void ClipStabilize::slotStabilizeFinished(bool success) +void ClipStabilize::fillParameters(QStringList lst) { - buttonBox->button(QDialogButtonBox::Abort)->setText(i18n("Close")); - button_start->setEnabled(true); - m_duration = 0; - if (success) { - log_text->setHtml(log_text->toPlainText() + "
" + i18n("Stabilize finished.")); - if (auto_add->isChecked()) { - KUrl url; - if (urls_list->count() > 0) { - url = KUrl(dest_url->url()); - } else url = dest_url->url(); - emit addClip(url); + m_ui_params.clear(); + while (!lst.isEmpty()){ + QString vallist=lst.takeFirst(); + QStringList cont=vallist.split(','); + QString name=cont.takeFirst(); + while (!cont.isEmpty()){ + QString valname=cont.takeFirst(); + QString val; + if (!cont.isEmpty()){ + val=cont.takeFirst(); + } + m_ui_params[name][valname]=val; } - if (urls_list->count() > 0 && m_urls.count() > 0) { - slotStartStabilize(); - return; - } else if (auto_close->isChecked()) accept(); - } else { - log_text->setHtml(log_text->toPlainText() + "
" + i18n("Stabilizing FAILED!")); } } +void ClipStabilize::slotValidate() +{ + if (m_urls.count() == 1) { + if (QFile::exists(dest_url->url().path())) { + if (KMessageBox::questionYesNo(this, i18n("File %1 already exists.\nDo you want to overwrite it?", dest_url->url().path() )) == KMessageBox::No) return; + } + } + else { + KUrl folder(dest_url->url()); + QStringList existingFiles; + foreach(const QString &path, m_urls) { + KUrl dest = folder; + dest.addPath(KUrl(path).fileName()); + if (QFile::exists(dest.path() + ".mlt")) existingFiles.append(dest.path() + ".mlt"); + } + if (!existingFiles.isEmpty()) { + if (KMessageBox::warningContinueCancelList(this, i18n("The stabilize job will overwrite the following files:"), existingFiles) == KMessageBox::Cancel) return; + } + } + accept(); +} #include "clipstabilize.moc"