]> git.sesse.net Git - kdenlive/blobdiff - src/smallruler.cpp
Integrate with the required MLT hooks for getting Movit to work.
[kdenlive] / src / smallruler.cpp
index 73243e0b0f6fc02b14198a294b47d63bcec3c5e8..d5b99032b95fa3b8b4cb6461491193260fde748a 100644 (file)
  ***************************************************************************/
 
 
+#include "smallruler.h"
+#include "kdenlivesettings.h"
+
+#include <KDebug>
+#include <KColorScheme>
+#include <KLocalizedString>
+
 #include <QMouseEvent>
 #include <QStylePainter>
 
-#include <KDebug>
+#define SEEK_INACTIVE (-1)
 
-#define LITTLE_MARK_X2 8
-#define LITTLE_MARK_X1 5
-#define MIDDLE_MARK_X1 3
-#define MIDDLE_MARK_X2 8
 
-#define LABEL_SIZE 8
+SmallRuler::SmallRuler(Monitor *monitor, Render *render, QWidget *parent) :
+        QWidget(parent)
+        ,m_cursorFramePosition(0)
+        ,m_maxval(2)
+        ,m_monitor(monitor)
+       ,m_render(render)
+       ,m_lastSeekPosition(SEEK_INACTIVE)
+       ,m_cursorColor(palette().text())
+{
+    m_zoneStart = 10;
+    m_zoneEnd = 60;
+    KSharedConfigPtr config = KSharedConfig::openConfig(KdenliveSettings::colortheme());
+    m_zoneColor = KStatefulBrush(KColorScheme::View, KColorScheme::FocusColor, config).brush(this).color();
+    m_zoneColor.setAlpha(180);
+
+    setMouseTracking(true);
+    setMinimumHeight(QFontInfo(font()).pixelSize());
+    adjustScale(m_maxval);
+}
 
-#include "smallruler.h"
 
+void SmallRuler::adjustScale(int maximum)
+{
+    m_maxval = maximum;
+    m_scale = (double) width() / (double) maximum;
+    if (m_scale == 0) m_scale = 1;
 
-SmallRuler::SmallRuler(QWidget *parent)
-        : KRuler(parent) {
-    setShowPointer(true);
-    setShowBigMarks(false);
-    setShowTinyMarks(false);
-    slotNewOffset(0);
-    setRulerMetricStyle(KRuler::Custom);
-    setLengthFixed(true);
-}
-
-void SmallRuler::setPixelPerMark(double rate) {
-    kDebug() << " RULER SET RATE: " << rate;
-    if (rate > 0.5) {
-        setLittleMarkDistance(25);
-        setMediumMarkDistance(5 * 25);
-    } else if (rate > 0.09) {
-        setLittleMarkDistance(5 * 25);
-        setMediumMarkDistance(30 * 25);
+    if (m_scale > 0.5) {
+        m_small = 25;
+        m_medium = 5 * 25;
+    } else if (m_scale > 0.09) {
+        m_small = 5 * 25;
+        m_medium = 30 * 25;
     } else {
-        setLittleMarkDistance(30 * 25);
-        setMediumMarkDistance(60 * 25);
+        m_small = 30 * 25;
+        m_medium = 60 * 25;
+    }
+    updatePixmap();
+}
+
+void SmallRuler::setZoneStart()
+{
+    int pos = m_render->requestedSeekPosition;
+    if (pos == SEEK_INACTIVE) pos = m_render->seekFramePosition();
+    setZone(pos, -1);
+}
+
+void SmallRuler::setZoneEnd()
+{
+    int pos = m_render->requestedSeekPosition;
+    if (pos == SEEK_INACTIVE) pos = m_render->seekFramePosition();
+    setZone(-1, pos);
+}
+
+void SmallRuler::setZone(int start, int end)
+{
+    if (start != -1) {
+        if (end != -1 && start >= end) {
+            m_zoneEnd = qMin(m_maxval, end + (start - m_zoneStart));
+            m_zoneStart = start;
+        } else if (end == -1 && start >= m_zoneEnd) {
+            m_zoneEnd = qMin(m_maxval, m_zoneEnd + (start - m_zoneStart));
+            m_zoneStart = start;
+        } else m_zoneStart = start;
+    }
+    if (end != -1) {
+        if (start != -1 && end <= start) {
+            m_zoneStart = qMax(0, start - (m_zoneEnd - end));
+            m_zoneEnd = end;
+        } else if (start == -1 && end <= m_zoneStart) {
+            m_zoneStart = qMax(0, m_zoneStart - (m_zoneEnd - end));
+            m_zoneEnd = end;
+        } else m_zoneEnd = end;
     }
+    updatePixmap();
+}
 
-    KRuler::setPixelPerMark(rate);
+void SmallRuler::setMarkers(const QList<CommentedTime> &list)
+{
+    m_markers = list;
+    updatePixmap();
+}
+
+QPoint SmallRuler::zone() const
+{
+    return QPoint(m_zoneStart, m_zoneEnd);
 }
 
 // virtual
-void SmallRuler::mousePressEvent(QMouseEvent * event) {
-    double pos = event->x() / pixelPerMark();
-    emit seekRenderer((int) pos);
+void SmallRuler::mousePressEvent(QMouseEvent * event)
+{
+    m_render->setActiveMonitor();
+    const int pos = event->x() / m_scale;
+    if (event->button() == Qt::RightButton) {
+        // Right button clicked, move selection zone
+        if (qAbs(pos - m_zoneStart) < qAbs(pos - m_zoneEnd)) m_zoneStart = pos;
+        else m_zoneEnd = pos;
+        emit zoneChanged(QPoint(m_zoneStart, m_zoneEnd));
+        updatePixmap();
+
+    } else if (pos != m_lastSeekPosition && pos != m_cursorFramePosition) {
+       m_render->seekToFrame(pos);
+       m_lastSeekPosition = pos;
+       update();
+    }
+    event->accept();
 }
 
 // virtual
-void SmallRuler::mouseMoveEvent(QMouseEvent * event) {
-    double pos = event->x() / pixelPerMark();
-    emit seekRenderer((int) pos);
+void SmallRuler::mouseReleaseEvent(QMouseEvent * event)
+{
+    event->accept();
 }
 
-void SmallRuler::slotNewValue(int _value) {
-    m_cursorPosition = (int)(_value);  /// pixelPerMark());
-    KRuler::slotNewValue(_value * pixelPerMark());
+
+// virtual
+void SmallRuler::leaveEvent(QEvent * event)
+{
+    QWidget::leaveEvent(event);
+    if (m_cursorColor == palette().link()) {
+       m_cursorColor = palette().text();
+       update();
+    }
 }
 
 // virtual
-void SmallRuler::paintEvent(QPaintEvent *e) {
-    //  debug ("KRuler::drawContents, %s",(horizontal==dir)?"horizontal":"vertical");
-
-    QStylePainter p(this);
-    p.fillRect(e->rect(), QBrush(QColor(Qt::white)));
-
-
-    int value  = this->value(),
-                 minval = minimum(),
-                          maxval;
-    maxval = maximum()
-             + offset() - endOffset();
-
-    //ioffsetval = value-offset;
-    //    pixelpm = (int)ppm;
-    //    left  = clip.left(),
-    //    right = clip.right();
-    double f, fend,
-    offsetmin = (double)(minval - offset()),
-                offsetmax = (double)(maxval - offset()),
-                            fontOffset = (((double)minval) > offsetmin) ? (double)minval : offsetmin;
-
-    // draw labels
-    QFont font = p.font();
-    font.setPointSize(LABEL_SIZE);
-    p.setFont(font);
-
-    if (showLittleMarks()) {
-        // draw the little marks
-        fend = pixelPerMark() * littleMarkDistance();
-        if (fend > 2) for (f = offsetmin; f < offsetmax; f += fend) {
-                p.drawLine((int)f, LITTLE_MARK_X1, (int)f, LITTLE_MARK_X2);
-            }
+void SmallRuler::mouseMoveEvent(QMouseEvent * event)
+{
+    const int pos = event->x() / m_scale;
+    if (event->buttons() & Qt::LeftButton) {
+       if (pos != m_lastSeekPosition && pos != m_cursorFramePosition) {
+           m_render->seekToFrame(pos);
+           m_lastSeekPosition = pos;
+           update();
+       }
+    }
+    else {
+       if (m_cursorColor == palette().text() && qAbs(pos - m_cursorFramePosition) * m_scale < 7) {
+           // Mouse is over cursor
+           m_cursorColor = palette().link();
+           update();
+       }
+       else if (m_cursorColor == palette().link() && qAbs(pos - m_cursorFramePosition) * m_scale >= 7) {
+           m_cursorColor = palette().text();
+           update();
+       }
+        if (qAbs((pos - m_zoneStart) * m_scale) < 4) {
+            setToolTip(i18n("Zone start: %1", m_monitor->getTimecodeFromFrames(m_zoneStart)));
+        } else if (qAbs((pos - m_zoneEnd) * m_scale) < 4) {
+            setToolTip(i18n("Zone end: %1", m_monitor->getTimecodeFromFrames(m_zoneEnd)));
+        } 
+       for (int i = 0; i < m_markers.count(); ++i) {
+           if (qAbs((pos - m_markers.at(i).time().frames(m_monitor->fps())) * m_scale) < 4) {
+               // We are on a marker
+               QString markerxt = m_monitor->getMarkerThumb(m_markers.at(i).time());
+               if (!markerxt.isEmpty()) {
+                   markerxt.prepend("<img src=\"");
+                   markerxt.append("\"><p align=\"center\">");
+               }
+               markerxt.append(m_markers.at(i).comment());
+               setToolTip(markerxt);
+               event->accept();
+               return;
+           }
+       }
+       if (pos > m_zoneStart && pos < m_zoneEnd) {
+            setToolTip(i18n("Zone duration: %1", m_monitor->getTimecodeFromFrames(m_zoneEnd - m_zoneStart)));
+       }
+       else setToolTip(i18n("Position: %1", m_monitor->getTimecodeFromFrames(pos)));
+    }
+    event->accept();
+}
+
+void SmallRuler::refreshRuler()
+{
+    m_lastSeekPosition = SEEK_INACTIVE;
+    update();
+}
+
+bool SmallRuler::slotNewValue(int value)
+{
+    if (value == m_cursorFramePosition) return false;
+    if (value == m_lastSeekPosition) m_lastSeekPosition = SEEK_INACTIVE;
+    m_cursorFramePosition = value;
+    /*int oldPos = m_cursorPosition;
+    m_cursorPosition = value * m_scale;
+    const int offset = 6;
+    const int x = qMin(oldPos, m_cursorPosition);
+    const int w = qAbs(oldPos - m_cursorPosition);
+    update(x - offset, 0, w + 2 * offset, height());*/
+    update();
+    return true;
+}
+
+
+//virtual
+void SmallRuler::resizeEvent(QResizeEvent *)
+{
+    adjustScale(m_maxval);
+}
+
+void SmallRuler::updatePixmap()
+{
+    m_pixmap = QPixmap(width(), height());
+    m_pixmap.fill(palette().window().color());
+    m_lastSeekPosition = SEEK_INACTIVE;
+    QPainter p(&m_pixmap);
+    double f, fend;
+
+    const int zoneStart = (int)(m_zoneStart * m_scale);
+    const int zoneEnd = (int)(m_zoneEnd * m_scale);
+    p.fillRect(zoneStart, 0, zoneEnd - zoneStart, height(), QBrush(m_zoneColor));
+
+    // draw ruler
+    p.setPen(palette().midlight().color());
+    //p.drawLine(0, 0, width(), 0);
+    p.drawLine(0, height() - 1, width(), height() - 1);
+    p.setPen(palette().dark().color());
+    // draw the little marks
+    fend = m_scale * m_small;
+    if (fend > 2) for (f = 0; f < width(); f += fend) {
+        p.drawLine((int)f, 0, (int)f, 3);
+    }
+
+    // draw medium marks
+    fend = m_scale * m_medium;
+    if (fend > 2) for (f = 0; f < width(); f += fend) {
+       p.drawLine((int)f, 0, (int)f, 6);
     }
-    if (showMediumMarks()) {
-        // draw medium marks
-        fend = pixelPerMark() * mediumMarkDistance();
-        if (fend > 2) for (f = offsetmin; f < offsetmax; f += fend) {
-                p.drawLine((int)f, MIDDLE_MARK_X1, (int)f, MIDDLE_MARK_X2);
-            }
+    // draw markers
+    if (!m_markers.isEmpty()) {
+        for (int i = 0; i < m_markers.count(); ++i) {
+           int pos = m_markers.at(i).time().frames(m_monitor->fps()) * m_scale;
+           p.setPen(CommentedTime::markerColor(m_markers.at(i).markerType()));
+            p.drawLine(pos, 0, pos, 9);
+        }
     }
+    p.end();
+    update();
+}
+
+// virtual
+void SmallRuler::paintEvent(QPaintEvent *e)
+{
 
-    /*   if (d->showem) {
-         // draw end marks
-         if (d->dir == Qt::Horizontal) {
-           p.drawLine(minval-d->offset, END_MARK_X1, minval-d->offset, END_MARK_X2);
-           p.drawLine(maxval-d->offset, END_MARK_X1, maxval-d->offset, END_MARK_X2);
-         }
-         else {
-           p.drawLine(END_MARK_X1, minval-d->offset, END_MARK_X2, minval-d->offset);
-           p.drawLine(END_MARK_X1, maxval-d->offset, END_MARK_X2, maxval-d->offset);
-         }
-       }*/
+    QPainter p(this);
+    QRect r = e->rect();
+    p.setClipRect(r);
+    p.drawPixmap(QPointF(), m_pixmap);
+    p.setPen(palette().shadow().color());
+    //p.setRenderHint(QPainter::Antialiasing, true);
+    //p.drawRoundedRect(rect().adjusted(1,1,-2,-2), 3, 3);
 
+    int cursorPos = m_cursorFramePosition * m_scale;
     // draw pointer
-    if (showPointer()) {
-        QPolygon pa(3);
-        pa.setPoints(3, value - 6, 0, value + 6, 0, value/*+0*/, 8);
-        p.setBrush(QBrush(Qt::yellow));
-        p.drawPolygon(pa);
+    QPolygon pa(3);
+    pa.setPoints(3, cursorPos - 6, height() - 1, cursorPos + 6, height() - 1, cursorPos/*+0*/, height() - 8);
+    p.setBrush(m_cursorColor);
+    p.setPen(Qt::NoPen);
+    p.drawPolygon(pa);
+
+    // Draw seeking pointer
+    if (m_lastSeekPosition != SEEK_INACTIVE && m_lastSeekPosition != m_cursorFramePosition) {
+       p.fillRect(m_lastSeekPosition * m_scale - 1, 0, 3, height(), palette().linkVisited());
     }
+}
 
+void SmallRuler::updatePalette()
+{
+    KSharedConfigPtr config = KSharedConfig::openConfig(KdenliveSettings::colortheme());
+    m_zoneColor = KStatefulBrush(KColorScheme::View, KColorScheme::FocusColor, config).brush(this).color();
+    m_zoneColor.setAlpha(180);
+    updatePixmap();
 }
 
 #include "smallruler.moc"