#include <QMouseEvent>
#include <QPaintEvent>
#include <QPainter>
#include "toonzqt/marksbar.h"
//*****************************************************************************
// Local namespace stuff
//*****************************************************************************
namespace {
void crop(QVector<int> &values, int min, int max) {
QVector<int>::iterator it, end = values.end();
for (it = values.begin(); it != end; ++it) *it = tcrop(*it, min, max);
}
//-----------------------------------------------------------------------------
void rollDown(QVector<int> &values, int max, int sortDist) {
assert(!values.empty());
values.back() = std::min(max, values.back());
int val;
QVector<int>::iterator it, jt, beg = values.begin();
for (it = values.end(), jt = --it; jt != beg; jt = it) {
--it;
val = *jt - sortDist;
if (*it > val) *it = val;
}
}
//-----------------------------------------------------------------------------
void rollUp(QVector<int> &values, int min, int sortDist) {
assert(!values.empty());
values.front() = std::max(min, values.front());
int val;
QVector<int>::iterator it, jt, end = values.end();
for (jt = values.begin(), it = jt++; jt != end; it = jt, ++jt) {
val = *it + sortDist;
if (val > *jt) *jt = val;
}
}
}
//*****************************************************************************
// MarksBar implementation
//*****************************************************************************
MarksBar::MarksBar(QWidget *parent, bool markUp)
: QFrame(parent)
, m_markUp(markUp)
, m_min(0)
, m_max(100)
, m_sortDist(-1)
, m_selectedMark(-1) {
setMinimumWidth(100);
setFixedHeight(6);
}
//-----------------------------------------------------------------------------
void MarksBar::setRange(int min, int max, int sortDist) {
assert(min <= max);
m_min = min, m_max = max, m_sortDist = sortDist;
conformValues();
update();
}
//-----------------------------------------------------------------------------
int MarksBar::valToPos(int val) {
const QRect contsRect = contentsRect();
return contsRect.left() +
contsRect.width() * ((val - m_min) / (double)(m_max - m_min));
}
//-----------------------------------------------------------------------------
int MarksBar::posToVal(int pos) {
const QRect contsRect = contentsRect();
return m_min +
(m_max - m_min) *
((pos - contsRect.left()) / (double)contsRect.width());
}
//-----------------------------------------------------------------------------
void MarksBar::conformValues(bool preferRollDown) {
if (m_values.empty()) return;
if (m_sortDist < 0)
::crop(m_values, m_min, m_max);
else {
if (preferRollDown) {
::rollDown(m_values, m_max, m_sortDist);
::rollUp(m_values, m_min, m_sortDist);
} else {
::rollUp(m_values, m_min, m_sortDist);
::rollDown(m_values, m_max, m_sortDist);
}
}
update();
emit marksUpdated();
}
//-----------------------------------------------------------------------------
void MarksBar::mousePressEvent(QMouseEvent *me) {
int newDist, currDist = (std::numeric_limits<int>::max)();
int val = posToVal(me->x());
// Select the nearest mark. If the distance is too great, select none.
int i, size = m_values.size();
for (i = 0; i < size; ++i) {
newDist = abs(m_values[i] - val);
if (newDist < 7 && newDist < currDist) m_selectedMark = i;
}
update();
}
//-----------------------------------------------------------------------------
void MarksBar::mouseMoveEvent(QMouseEvent *me) {
if (m_selectedMark < 0) return;
int newVal = tcrop(posToVal(me->x()), m_min, m_max);
bool left = newVal < m_values[m_selectedMark];
m_values[m_selectedMark] = newVal;
conformValues(left);
update();
}
//-----------------------------------------------------------------------------
void MarksBar::mouseReleaseEvent(QMouseEvent *me) {
m_selectedMark = -1;
emit marksReleased();
}
//-----------------------------------------------------------------------------
void MarksBar::drawMark(QPainter &p, int pos, const QColor &color) {
QPolygon poly(3);
if (m_markUp) {
poly[0] = QPoint(pos - 5, 5);
poly[1] = QPoint(pos + 5, 5);
poly[2] = QPoint(pos, 0);
} else {
poly[0] = QPoint(pos - 5, 0);
poly[1] = QPoint(pos + 5, 0);
poly[2] = QPoint(pos, 5);
}
p.setBrush(color);
p.drawPolygon(poly);
}
//-----------------------------------------------------------------------------
void MarksBar::paintEvent(QPaintEvent *pe) {
QPainter p(this);
assert(m_values.size() == m_colors.size());
// Draw markers
int i, size = m_values.size();
for (i = 0; i < size; ++i) drawMark(p, valToPos(m_values[i]), m_colors[i]);
}