summaryrefslogtreecommitdiffstats
path: root/src/base/SnapGrid.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/base/SnapGrid.cpp')
-rw-r--r--src/base/SnapGrid.cpp192
1 files changed, 192 insertions, 0 deletions
diff --git a/src/base/SnapGrid.cpp b/src/base/SnapGrid.cpp
new file mode 100644
index 0000000..6d0061e
--- /dev/null
+++ b/src/base/SnapGrid.cpp
@@ -0,0 +1,192 @@
+// -*- c-basic-offset: 4 -*-
+
+/*
+ Rosegarden
+ A sequencer and musical notation editor.
+
+ This program is Copyright 2000-2008
+ Guillaume Laurent <glaurent@telegraph-road.org>,
+ Chris Cannam <cannam@all-day-breakfast.com>,
+ Richard Bown <bownie@bownie.com>
+
+ The moral right of the authors to claim authorship of this work
+ has been asserted.
+
+ 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. See the file
+ COPYING included with this distribution for more information.
+*/
+
+#include "SnapGrid.h"
+#include "Composition.h"
+
+namespace Rosegarden {
+
+
+//////////////////////////////////////////////////////////////////////
+// SnapGrid
+//////////////////////////////////////////////////////////////////////
+
+
+const timeT SnapGrid::NoSnap = -1;
+const timeT SnapGrid::SnapToBar = -2;
+const timeT SnapGrid::SnapToBeat = -3;
+const timeT SnapGrid::SnapToUnit = -4;
+
+SnapGrid::SnapGrid(RulerScale *rulerScale, int ysnap) :
+ m_rulerScale(rulerScale),
+ m_snapTime(SnapToBeat),
+ m_ysnap(ysnap)
+{
+ // nothing else
+}
+
+void
+SnapGrid::setSnapTime(timeT snap)
+{
+ assert(snap > 0 ||
+ snap == NoSnap ||
+ snap == SnapToBar ||
+ snap == SnapToBeat ||
+ snap == SnapToUnit);
+ m_snapTime = snap;
+}
+
+timeT
+SnapGrid::getSnapSetting() const
+{
+ return m_snapTime;
+}
+
+timeT
+SnapGrid::getSnapTime(double x) const
+{
+ timeT time = m_rulerScale->getTimeForX(x);
+ return getSnapTime(time);
+}
+
+timeT
+SnapGrid::getSnapTime(timeT time) const
+{
+ if (m_snapTime == NoSnap) return 0;
+
+ Rosegarden::Composition *composition = m_rulerScale->getComposition();
+ int barNo = composition->getBarNumber(time);
+ std::pair<timeT, timeT> barRange = composition->getBarRange(barNo);
+
+ timeT snapTime = barRange.second - barRange.first;
+
+ if (m_snapTime == SnapToBeat) {
+ snapTime = composition->getTimeSignatureAt(time).getBeatDuration();
+ } else if (m_snapTime == SnapToUnit) {
+ snapTime = composition->getTimeSignatureAt(time).getUnitDuration();
+ } else if (m_snapTime != SnapToBar && m_snapTime < snapTime) {
+ snapTime = m_snapTime;
+ }
+
+ return snapTime;
+}
+
+timeT
+SnapGrid::snapX(double x, SnapDirection direction) const
+{
+ return snapTime(m_rulerScale->getTimeForX(x), direction);
+}
+
+timeT
+SnapGrid::snapTime(timeT time, SnapDirection direction) const
+{
+ if (m_snapTime == NoSnap) return time;
+
+ Rosegarden::Composition *composition = m_rulerScale->getComposition();
+ int barNo = composition->getBarNumber(time);
+ std::pair<timeT, timeT> barRange = composition->getBarRange(barNo);
+
+ timeT snapTime = barRange.second - barRange.first;
+
+ if (m_snapTime == SnapToBeat) {
+ snapTime = composition->getTimeSignatureAt(time).getBeatDuration();
+ } else if (m_snapTime == SnapToUnit) {
+ snapTime = composition->getTimeSignatureAt(time).getUnitDuration();
+ } else if (m_snapTime != SnapToBar && m_snapTime < snapTime) {
+ snapTime = m_snapTime;
+ }
+
+ timeT offset = (time - barRange.first);
+ timeT rounded = (offset / snapTime) * snapTime;
+
+ timeT left = rounded + barRange.first;
+ timeT right = left + snapTime;
+
+ if (direction == SnapLeft) return left;
+ else if (direction == SnapRight) return right;
+ else if ((offset - rounded) > (rounded + snapTime - offset)) return right;
+ else return left;
+}
+
+int
+SnapGrid::getYBin(int y) const
+{
+ if (m_ysnap == 0) return y;
+
+ int cy = 0;
+
+ std::map<int, int>::const_iterator i = m_ymultiple.begin();
+
+ int nextbin = -1;
+ if (i != m_ymultiple.end()) nextbin = i->first;
+
+ for (int b = 0; ; ++b) {
+
+ if (nextbin == b) {
+
+ cy += i->second * m_ysnap;
+ ++i;
+ if (i == m_ymultiple.end()) nextbin = -1;
+ else nextbin = i->first;
+
+ } else {
+
+ cy += m_ysnap;
+ }
+
+ if (cy > y) {
+ return b;
+ }
+ }
+}
+
+int
+SnapGrid::getYBinCoordinate(int bin) const
+{
+ if (m_ysnap == 0) return bin;
+
+ int y = 0;
+
+ std::map<int, int>::const_iterator i = m_ymultiple.begin();
+
+ int nextbin = -1;
+ if (i != m_ymultiple.end()) nextbin = i->first;
+
+ for (int b = 0; b < bin; ++b) {
+
+ if (nextbin == b) {
+
+ y += i->second * m_ysnap;
+ ++i;
+ if (i == m_ymultiple.end()) nextbin = -1;
+ else nextbin = i->first;
+
+ } else {
+
+ y += m_ysnap;
+ }
+ }
+
+ return y;
+}
+
+
+}