Logo Search packages:      
Sourcecode: kdenlive version File versions

transition.cpp

/***************************************************************************
                          transition.cpp  -  description
                             -------------------
    begin                : Tue Jan 24 2006
    copyright            : (C) 2006 by Jean-Baptiste Mardelle
    email                : jb@ader.ch 
 ***************************************************************************/

/***************************************************************************
 *                                                                         *
 *   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.                                   *
 *                                                                         *
 ***************************************************************************/

#include "transition.h"
#include "docclipref.h"
#include "kdenlivesettings.h"

#include <kdebug.h>
#include <kiconloader.h>
#include <klocale.h>
#include <qdom.h>

/* Transitions can be either be
    1) placed on a track. They are not dependant of any clip // not implemented yet
    2) Attached to one clip. They move with the clip
    3) Attached to two clips. They automatically adjust to the length of the overlapping area //disabled for the moment
*/

/* create an "automatic duration" transition */
Transition::Transition(const DocClipRef * clipa, const DocClipRef * clipb, const QString &type)
{
    m_invertTransition = false;
    m_singleClip = true;
    m_transitionTrack = 0;
    m_secondClip = NULL;
    m_transitionType = getTransitionForName(type);
    m_transitionName = getTransitionName(m_transitionType);

    if (clipb) {
        // Transition is an automatic transition between 2 clips
        m_singleClip = true;
       /* if (clipa->trackNum()>clipb->trackNum()) {
            m_referenceClip = clipb;
            m_secondClip = clipa;
        }
        else {
            m_referenceClip = clipa;
            m_secondClip = clipb;
        }*/

      m_referenceClip = clipa;
      m_secondClip = clipb;
    
        if (m_referenceClip->trackStart() < m_secondClip->trackStart())
            m_invertTransition = true;
      else if (m_referenceClip->trackStart() == m_secondClip->trackStart() && m_referenceClip->trackEnd() < m_secondClip->trackEnd())
            m_invertTransition = true;

      GenTime startb = m_secondClip->trackStart();
        GenTime starta = m_referenceClip->trackStart();
      GenTime endb = m_secondClip->trackEnd();
        GenTime enda = m_referenceClip->trackEnd();
      GenTime transitionDuration;
      if (starta >= startb && starta <= endb) {
            m_transitionStart = GenTime(0.0);
            if (endb < enda) m_transitionDuration = endb - starta;
            else m_transitionDuration = enda - starta;
      }
      else if (startb >= starta && startb <= enda){
            m_transitionStart = startb - starta;
            if (endb > enda) m_transitionDuration = enda - startb;
            else m_transitionDuration = endb - startb;
      }
      else {
            if (starta > endb)
                  m_transitionStart = GenTime(0.0);
            else m_transitionStart = enda - starta -GenTime(3.0);

            m_transitionDuration = GenTime(3.0);
      }

      if (m_transitionDuration < GenTime(0.12)) m_transitionDuration = GenTime(0.12);
      m_secondClip = 0; // disable auto transition for the moment
    }
    else {
        // Transition is attached to a single clip
        m_referenceClip = clipa;
        m_transitionStart = GenTime(0.0);
        
        // Default duration = 1,5 seconds
        if (GenTime(1.5) > m_referenceClip->cropDuration()) 
            m_transitionDuration = m_referenceClip->cropDuration();
        else m_transitionDuration = GenTime(1.5);
        m_secondClip = 0;
    }
}

/* create an "simple" transition (type 2) */
Transition::Transition(const DocClipRef * clipa)
{
    m_invertTransition = false;
    m_singleClip = true;
    m_transitionTrack = 0;
    m_secondClip = NULL;
    m_transitionType = LUMA_TRANSITION;
    m_transitionName = getTransitionName(m_transitionType);

        m_referenceClip = clipa;
        m_transitionStart = GenTime(0.0);
        
        // Default duration = 2.5 seconds
        if (GenTime(1.5) > m_referenceClip->cropDuration()) 
            m_transitionDuration = m_referenceClip->cropDuration();
        else m_transitionDuration = GenTime(2.5);
        m_secondClip = 0;
}

/* create an "simple" transition (type 2) around a given time*/
Transition::Transition(const DocClipRef * clipa, const GenTime &time, const QString &type)
{
    m_invertTransition = false;
    m_singleClip = true;
    m_transitionTrack = 0;
    m_secondClip = NULL;
    m_transitionType = getTransitionForName(type);
    m_transitionName = getTransitionName(m_transitionType);
    
    // Default duration = 2.5 seconds
    GenTime defaultTransitionDuration = GenTime(2.5);

    m_referenceClip = clipa;
    if (time - m_referenceClip->trackStart() < GenTime(2.0)) m_transitionStart = GenTime(0.0);
    else if (m_referenceClip->trackEnd() - time < GenTime(2.0)) m_transitionStart = m_referenceClip->cropDuration() - defaultTransitionDuration;
    else m_transitionStart = time - m_referenceClip->trackStart() - GenTime(1.0);
    
    if (m_transitionStart + defaultTransitionDuration > m_referenceClip->cropDuration()) 
        m_transitionDuration = m_referenceClip->cropDuration() - m_transitionStart;
    else m_transitionDuration = defaultTransitionDuration;
    
    if (time > clipa->trackMiddleTime()) m_invertTransition = true;
    m_secondClip = 0;
}

/* create an "simple" transition (type 2) */
Transition::Transition(const DocClipRef * clipa, const TRANSITIONTYPE & type, const GenTime &startTime, const GenTime &endTime, bool inverted)
{
    m_invertTransition = inverted;
    m_singleClip = true;
    m_transitionTrack = 0;
    m_secondClip = NULL;
    m_transitionType = type;
    m_transitionName = getTransitionName(m_transitionType);

    GenTime duration = endTime - startTime;
    
    // Default duration = 2.5 seconds
    GenTime defaultTransitionDuration = GenTime(2.5);
    
    m_referenceClip = clipa;
    if (startTime < m_referenceClip->trackStart()) m_transitionStart = GenTime(0.0);
    else if (startTime > m_referenceClip->trackEnd()) m_transitionStart = m_referenceClip->cropDuration() - defaultTransitionDuration;
    else m_transitionStart = startTime - m_referenceClip->trackStart();
    
    if (m_transitionStart + duration > m_referenceClip->cropDuration()) 
        m_transitionDuration = m_referenceClip->cropDuration() - m_transitionStart;
    else m_transitionDuration = duration;
    m_secondClip = 0;
}

// create a transition from XML
Transition::Transition(const DocClipRef * clip, QDomElement transitionElement, GenTime offset)
{
      if (offset == GenTime()) offset = clip->trackStart();
      m_referenceClip = clip;
      m_singleClip = true;
        m_secondClip = NULL;
      m_transitionStart = GenTime (transitionElement.attribute("start", QString::null).toInt(),KdenliveSettings::defaultfps());
      m_transitionDuration = GenTime(transitionElement.attribute("end", QString::null).toInt(),KdenliveSettings::defaultfps()) - m_transitionStart;
      m_transitionTrack = transitionElement.attribute("transition_track", "0").toInt();
      m_transitionStart = m_transitionStart - offset;

        m_invertTransition = transitionElement.attribute("inverted", "0").toInt();
      uint transType = transitionElement.attribute("type", "0").toInt();
      if (transType == LUMA_TRANSITION) m_transitionType = LUMA_TRANSITION;
      else if (transType == COMPOSITE_TRANSITION) m_transitionType = COMPOSITE_TRANSITION; 
      else if (transType == PIP_TRANSITION) m_transitionType = PIP_TRANSITION;
      else if (transType == LUMAFILE_TRANSITION) m_transitionType = LUMAFILE_TRANSITION;
      else if (transType == MIX_TRANSITION) m_transitionType = MIX_TRANSITION;

      // load transition parameters
        typedef QMap<QString, QString> ParamMap;
        ParamMap params;
        for( QDomNode n = transitionElement.firstChild(); !n.isNull(); n = n.nextSibling() )
        {
            QDomElement paramElement = n.toElement();
            params[paramElement.tagName()] = paramElement.attribute("value", QString::null);
        }
        if (!params.isEmpty()) setTransitionParameters(params);
      
      // Check if transition is valid (not outside of clip)
      if (m_transitionStart > clip->cropDuration()) 
            m_transitionDuration = GenTime();
}

Transition::~Transition()
{
}

void Transition::setTransitionType(TRANSITIONTYPE newType)
{
    m_transitionType = newType;
    m_transitionName = getTransitionName(m_transitionType);
}

Transition::TRANSITIONTYPE Transition::transitionType()
{
    return m_transitionType;
}

QString Transition::transitionTag()
{
    switch (m_transitionType) {
      case COMPOSITE_TRANSITION:
          return "composite";
      case PIP_TRANSITION:
          return "composite";
      case MIX_TRANSITION:
          return "mix";
      default:
          return "luma";

    }
}

QString Transition::getTransitionName(const TRANSITIONTYPE & type)
{
    if (type == COMPOSITE_TRANSITION) return i18n("Push");
    else if (type == PIP_TRANSITION) return i18n("Pip");
    else if (type == LUMAFILE_TRANSITION) return i18n("Wipe");
    else if (type == MIX_TRANSITION) return i18n("Audio Fade");
    return i18n("Crossfade");
}

Transition::TRANSITIONTYPE Transition::getTransitionForName(const QString & type)
{
    if (type == i18n("Push")) return COMPOSITE_TRANSITION;
    else if (type == i18n("Pip")) return PIP_TRANSITION;
    else if (type == i18n("Wipe")) return LUMAFILE_TRANSITION;
    return LUMA_TRANSITION;
}


QString Transition::transitionName()
{
    return m_transitionName;
}

void Transition::setTransitionParameters(const QMap < QString, QString > parameters)
{
    m_transitionParameters = parameters;
}

const QMap < QString, QString > Transition::transitionParameters()
{
    return m_transitionParameters;
}

bool Transition::invertTransition()
{
    if (!m_singleClip) {
        if (m_referenceClip->trackStart() < m_secondClip->trackStart()) return true;
        else return false;
    }
    return m_invertTransition;
}

QPixmap Transition::transitionPixmap()
{
    if (m_transitionType == LUMA_TRANSITION) {
      if (invertTransition()) return KGlobal::iconLoader()->loadIcon("kdenlive_trans_down", KIcon::Small, 15);
      else return KGlobal::iconLoader()->loadIcon("kdenlive_trans_up", KIcon::Small, 15);
    }
    else if (m_transitionType == COMPOSITE_TRANSITION) {
         return KGlobal::iconLoader()->loadIcon("kdenlive_trans_wiper", KIcon::Small, 15);
    }
    else return KGlobal::iconLoader()->loadIcon("kdenlive_trans_pip", KIcon::Small, 15);
}

int Transition::transitionTrack()
{
    return m_transitionTrack;
}

void Transition::setTransitionTrack(int track)
{
    m_transitionTrack = track;
}

void Transition::setTransitionDirection(bool inv)
{
    m_invertTransition = inv;
}

int Transition::transitionDocumentTrack()
{
    return m_referenceClip->trackNum();
}

int Transition::transitionStartTrack()
{
    return m_referenceClip->playlistTrackNum();
}

int Transition::transitionEndTrack()
{
    if (!m_singleClip) return m_secondClip->playlistTrackNum();
    if (m_transitionTrack == 0) return m_referenceClip->playlistNextTrackNum();
    else if (m_transitionTrack == 1) return 0;
    else return m_referenceClip->playlistOtherTrackNum(m_transitionTrack - 1);
}

GenTime Transition::transitionStartTime()
{
    if (!m_singleClip) {
        GenTime startb = m_secondClip->trackStart();
        GenTime starta = m_referenceClip->trackStart();
        if (startb > m_referenceClip->trackEnd()) return m_referenceClip->trackEnd() - GenTime(0.12);
        if (startb > starta)
         return startb;
        return starta;
    }
    else return m_referenceClip->trackStart() + m_transitionStart;
}


GenTime Transition::transitionEndTime()
{
    if (!m_singleClip) {
        GenTime endb = m_secondClip->trackEnd();
        GenTime enda = m_referenceClip->trackEnd();
        if (m_secondClip->trackStart() > enda) return enda;
        if (endb < m_referenceClip->trackStart()) return m_referenceClip->trackStart() + GenTime(0.12);
        else if (endb > enda) return enda;
        else return endb;
    }
    else { 
        if (m_transitionStart + m_transitionDuration > m_referenceClip->cropDuration())
            return m_referenceClip->trackEnd();
        return m_referenceClip->trackStart() + m_transitionStart + m_transitionDuration;
    }
}


void Transition::resizeTransitionStart(GenTime time)
{
    if (!m_singleClip) return; //cannot resize automatic transitions
    if (time < m_referenceClip->trackStart()) time = m_referenceClip->trackStart();
    // Transitions shouldn't be shorter than 3 frames, about 0.12 seconds
    if ( transitionEndTime().ms() - time.ms() < 120.0) time = transitionEndTime() - GenTime(0.12);
    m_transitionDuration =m_transitionDuration - (time - m_referenceClip->trackStart() - m_transitionStart);
    m_transitionStart = time - m_referenceClip->trackStart();
}

void Transition::resizeTransitionEnd(GenTime time)
{
    if (!m_singleClip) return; //cannot resize automatic transitions
    if (time > m_referenceClip->trackEnd()) time = m_referenceClip->trackEnd();
    // Transitions shouldn't be shorter than 3 frames, about 0.12 seconds
    if ( time.ms() - transitionStartTime().ms() < 120.0) time = transitionStartTime() + GenTime(0.12);
    m_transitionDuration = time - ( m_referenceClip->trackStart() + m_transitionStart);
}

void Transition::moveTransition(GenTime time)
{
    if (!m_singleClip) return; //cannot move automatic transitions
    if (m_transitionStart + time < GenTime(0.0)) m_transitionStart = GenTime(0.0);
    else if ( m_transitionStart + time > m_referenceClip->cropDuration() - m_transitionDuration)
        m_transitionStart = m_referenceClip->cropDuration() - m_transitionDuration;
    else m_transitionStart = m_transitionStart + time;
    if (m_transitionStart < GenTime(0.0)) m_transitionStart = GenTime(0.0);
}

bool Transition::hasClip(const DocClipRef * clip)
{
    if (clip == m_secondClip) return true;
    return false;
}

bool Transition::belongsToClip(const DocClipRef * clip)
{
    if (clip == m_referenceClip) return true;
    return false;
}

Transition *Transition::clone()
{
    if (m_singleClip || m_secondClip == 0)
        return new Transition::Transition(m_referenceClip);
    else
        //return new Transition::Transition(m_referenceClip, m_secondClip);
      return new Transition::Transition(m_referenceClip, this->toXML());
}

Transition *Transition::reparent(const DocClipRef * clip)
{
    return new Transition::Transition(clip, this->toXML(), m_referenceClip->trackStart());
}

bool Transition::isValid()
{
    return (m_transitionDuration != GenTime());
}

QDomElement Transition::toXML()
{
    QDomDocument doc;
    QDomElement effect = doc.createElement("ktransition");
    effect.setAttribute("type", transitionType());
    effect.setAttribute("inverted", invertTransition());
    effect.setAttribute("transition_track", m_transitionTrack);
    effect.setAttribute("start", transitionStartTime().frames(KdenliveSettings::defaultfps()));
    effect.setAttribute("end", transitionEndTime().frames(KdenliveSettings::defaultfps()));
    if (m_secondClip) {
        effect.setAttribute("clipb_starttime", m_secondClip->trackStart().frames(KdenliveSettings::defaultfps()));
        effect.setAttribute("clipb_track", transitionEndTrack());
    }
    

    QMap<QString, QString>::Iterator it;
    for ( it = m_transitionParameters.begin(); it != m_transitionParameters.end(); ++it ) {
        QDomElement param = doc.createElement(it.key());
        param.setAttribute("value",it.data());
        effect.appendChild(param);
    }

    return effect;
}


Generated by  Doxygen 1.6.0   Back to index