Blob Blame Raw


#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]);
}