Skip to content

Commit

Permalink
Revert "Revert "Normalize potmeter widget values to [0.0, 1.0] instea…
Browse files Browse the repository at this point in the history
…d of [0.0, 127.0].""

This reverts commit 6fcd283.
  • Loading branch information
rryan committed Jan 3, 2014
1 parent 6fcd283 commit bc4188a
Show file tree
Hide file tree
Showing 14 changed files with 112 additions and 88 deletions.
52 changes: 31 additions & 21 deletions src/control/controlbehavior.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -54,35 +54,42 @@ double ControlPotmeterBehavior::valueToWidgetParameter(double dValue) {
} else if (dValue < m_dMinValue) {
dValue = m_dMinValue;
}
double dNorm = (dValue - m_dMinValue) / m_dValueRange;
return dNorm < 0.5 ? dNorm * 128.0 : dNorm * 126.0 + 1.0;
if (m_dValueRange == 0.0) {
return 0;
}
return (dValue - m_dMinValue) / m_dValueRange;
}

double ControlPotmeterBehavior::widgetParameterToValue(double dParam) {
double dNorm = dParam < 64 ? dParam / 128.0 : (dParam - 1.0) / 126.0;
return m_dMinValue + dNorm * m_dValueRange;
return m_dMinValue + dParam * m_dValueRange;
}

double ControlPotmeterBehavior::valueToMidiParameter(double dValue) {
return valueToWidgetParameter(dValue);
// 7-bit MIDI has 128 values [0, 127]. This means there is no such thing as
// center. The industry convention is that 64 is center. We fake things a
// little bit here to make that the case. This piece-wise function is linear
// from 0 to 64 with slope 128 and from 64 to 127 with slope 126.
double dNorm = valueToWidgetParameter(dValue);
return dNorm < 0.5 ? dNorm * 128.0 : dNorm * 126.0 + 1.0;
}

void ControlPotmeterBehavior::setValueFromMidiParameter(MidiOpCode o, double dParam,
ControlDoublePrivate* pControl) {
Q_UNUSED(o);
pControl->set(widgetParameterToValue(dParam), NULL);
double dNorm = dParam < 64 ? dParam / 128.0 : (dParam - 1.0) / 126.0;
pControl->set(widgetParameterToValue(dNorm), NULL);
}

#define maxPosition 127
#define minPosition 0
#define middlePosition ((maxPosition-minPosition)/2)
#define maxPosition 1.0
#define minPosition 0.0
#define middlePosition ((maxPosition-minPosition)/2.0)
#define positionrange (maxPosition-minPosition)

ControlLogpotmeterBehavior::ControlLogpotmeterBehavior(double dMaxValue)
: ControlPotmeterBehavior(0, dMaxValue) {
if (dMaxValue == 1.0) {
m_bTwoState = false;
m_dB1 = log10(2.0)/maxPosition;
m_dB1 = log10(2.0) / maxPosition;
} else {
m_bTwoState = true;
m_dB1 = log10(2.0) / middlePosition;
Expand Down Expand Up @@ -134,26 +141,29 @@ ControlLinPotmeterBehavior::ControlLinPotmeterBehavior(double dMinValue, double
ControlLinPotmeterBehavior::~ControlLinPotmeterBehavior() {
}

double ControlLinPotmeterBehavior::valueToWidgetParameter(double dValue) {
if (dValue > m_dMaxValue) {
dValue = m_dMaxValue;
} else if (dValue < m_dMinValue) {
dValue = m_dMinValue;
}
double dNorm = (dValue - m_dMinValue) / m_dValueRange;
return math_min(dNorm * 128, 127);
double ControlLinPotmeterBehavior::valueToMidiParameter(double dValue) {
// 7-bit MIDI has 128 values [0, 127]. This means there is no such thing as
// center. The industry convention is that 64 is center. We fake things a
// little bit here to make that the case. This function is linear from [0,
// 127.0/128.0] with slope 128 and then cuts off at 127 from 127.0/128.0 to
// 1.0. from 0 to 64 with slope 128 and from 64 to 127 with slope 126.
double dNorm = valueToWidgetParameter(dValue);
return math_max(127.0, dNorm * 128.0);
}

double ControlLinPotmeterBehavior::widgetParameterToValue(double dParam) {
void ControlLinPotmeterBehavior::setValueFromMidiParameter(MidiOpCode o, double dParam,
ControlDoublePrivate* pControl) {
Q_UNUSED(o);
double dNorm = dParam / 128.0;
return m_dMinValue + dNorm * m_dValueRange;
pControl->set(widgetParameterToValue(dNorm), NULL);
}

double ControlTTRotaryBehavior::valueToWidgetParameter(double dValue) {
return dValue * 200.0 + 64;
return (dValue * 200.0 + 64) / 127.0;
}

double ControlTTRotaryBehavior::widgetParameterToValue(double dParam) {
dParam *= 127.0;
// Non-linear scaling
double temp = ((dParam - 64.0) * (dParam - 64.0)) / 64.0;
if (dParam - 64 < 0) {
Expand Down
5 changes: 3 additions & 2 deletions src/control/controlbehavior.h
Original file line number Diff line number Diff line change
Expand Up @@ -65,8 +65,9 @@ class ControlLinPotmeterBehavior : public ControlPotmeterBehavior {
ControlLinPotmeterBehavior(double dMinValue, double dMaxValue);
virtual ~ControlLinPotmeterBehavior();

virtual double valueToWidgetParameter(double dValue);
virtual double widgetParameterToValue(double dParam);
virtual double valueToMidiParameter(double dValue);
virtual void setValueFromMidiParameter(MidiOpCode o, double dParam,
ControlDoublePrivate* pControl);
};

class ControlTTRotaryBehavior : public ControlNumericBehavior {
Expand Down
4 changes: 0 additions & 4 deletions src/controllinpotmeter.cpp
Original file line number Diff line number Diff line change
@@ -1,14 +1,10 @@
#include "controllinpotmeter.h"
#include "defs.h"

// This control has a linear link between the m_dValue and the Midi Value
// limitation: m_dMaxValue represents the midi value of 128 and is never reached
ControlLinPotmeter::ControlLinPotmeter(ConfigKey key, double dMinValue, double dMaxValue) :
ControlPotmeter(key, dMinValue, dMaxValue) {
if (m_pControl) {
m_pControl->setBehavior(
new ControlLinPotmeterBehavior(dMinValue, dMaxValue));
}
}


12 changes: 4 additions & 8 deletions src/controllogpotmeter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,16 +21,13 @@
Purpose: Creates a new logarithmic potmeter, where the value is
given by:
value = 10^(b*midibyte) - 1
value = 10^(b*parameter) - 1
The lower value is 0, for midibyte=64 the value is 1 and the upper
The lower value is 0, for parameter=0.5 the value is 1 and the upper
value is set by maxvalue.
If the maxvalue is set to 1, the potmeter operates with only
one logarithmid scale between 0 (for midi 0) and 1 (midivalue 128).
Input: n - name
midino - number of the midi controller.
midicontroller - pointer to the midi controller.
If the maxvalue is 1, the potmeter operates with only one
logarithmic scale between 0 (for parameter 0) and 1 (parameter 1.0).
-------- ------------------------------------------------------ */
ControlLogpotmeter::ControlLogpotmeter(ConfigKey key, double dMaxValue)
: ControlPotmeter(key, 0, dMaxValue) {
Expand All @@ -43,4 +40,3 @@ ControlLogpotmeter::ControlLogpotmeter(ConfigKey key, double dMaxValue)
new ControlLogpotmeterBehavior(dMaxValue));
}
}

4 changes: 2 additions & 2 deletions src/engine/enginebuffer.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -180,8 +180,8 @@ EngineBuffer::EngineBuffer(const char* _group, ConfigObject<ConfigValue>* _confi
m_visualBpm = new ControlObject(ConfigKey(m_group, "visual_bpm"));
m_visualKey = new ControlObject(ConfigKey(m_group, "visual_key"));

// Slider to show and change song position
//these bizarre choices map conveniently to the 0-127 range of midi
// Slider to show and change song position. kMinPlayposRange and
// kMaxPlayposRange map conveniently to the 0-127 range of 7-bit MIDI.
m_playposSlider = new ControlLinPotmeter(
ConfigKey(m_group, "playposition"), kMinPlayposRange, kMaxPlayposRange);
connect(m_playposSlider, SIGNAL(valueChanged(double)),
Expand Down
16 changes: 8 additions & 8 deletions src/widget/knobeventhandler.h
Original file line number Diff line number Diff line change
Expand Up @@ -31,13 +31,12 @@ class KnobEventHandler {
}

double value = pWidget->getValue();
value += dist;
// For legacy (MIDI) reasons this is tuned to 127.
value += dist / 127.0;
QCursor::setPos(m_startPos);

if (value > 127.0)
value = 127.0;
else if (value < 0.0)
value = 0.0;
// Clamp to [0.0, 1.0]
value = math_max(0.0, math_min(1.0, value));

pWidget->setValue(value);
emit(pWidget->valueChangedLeftDown(value));
Expand Down Expand Up @@ -80,11 +79,12 @@ class KnobEventHandler {
}

void wheelEvent(T* pWidget, QWheelEvent* e) {
double wheelDirection = e->delta() / 120.;
// For legacy (MIDI) reasons this is tuned to 127.
double wheelDirection = e->delta() / (120.0 * 127.0);
double newValue = pWidget->getValue() + wheelDirection;

// Clamp to [0.0, 127.0]
newValue = math_max(0.0, math_min(127.0, newValue));
// Clamp to [0.0, 1.0]
newValue = math_max(0.0, math_min(1.0, newValue));

pWidget->updateValue(newValue);
e->accept();
Expand Down
3 changes: 1 addition & 2 deletions src/widget/wdisplay.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -108,8 +108,7 @@ void WDisplay::setPixmap(QVector<PaintablePointer>* pPixmaps, int iPos,
}

int WDisplay::getActivePixmapIndex() const {
return static_cast<int>(
m_value * static_cast<double>(m_pixmaps.size()) / 128.0);
return static_cast<int>(m_value * m_pixmaps.size());
}

void WDisplay::paintEvent(QPaintEvent* ) {
Expand Down
4 changes: 1 addition & 3 deletions src/widget/wknob.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,9 +29,7 @@ WKnob::~WKnob() {
}

int WKnob::getActivePixmapIndex() const {
// TODO(rryan): Ew.
int iNoPos = numPixmaps();
return (int)(((m_value-64.)*(((float)iNoPos-1.)/127.))+((float)iNoPos/2.));
return static_cast<int>(m_value * numPixmaps());
}

void WKnob::mouseMoveEvent(QMouseEvent* e) {
Expand Down
4 changes: 2 additions & 2 deletions src/widget/wknobcomposed.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -71,8 +71,8 @@ void WKnobComposed::paintEvent(QPaintEvent* e) {
if (!m_pKnob.isNull() && !m_pKnob->isNull()) {
p.translate(width() / 2.0, height() / 2.0);

// Value is now in the range [0, 1].
double value = getValue() / 127.0;
// Value is in the range [0, 1].
double value = getValue();

double angle = m_dMinAngle + (m_dMaxAngle - m_dMinAngle) * value;
p.rotate(angle);
Expand Down
32 changes: 22 additions & 10 deletions src/widget/woverview.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -120,14 +120,14 @@ void WOverview::setup(QDomNode node) {
//qDebug() << "WOverview : m_markRanges" << m_markRanges.size();
}

void WOverview::setValue(double fValue) {
void WOverview::setValue(double dValue) {
if (!m_bDrag)
{
// Calculate handle position
int iPos = valueToPosition(fValue);
int iPos = valueToPosition(dValue);
if (iPos != m_iPos) {
m_iPos = iPos;
//qDebug() << "WOverview::setValue" << fValue << ">>" << m_iPos;
//qDebug() << "WOverview::setValue" << dValue << ">>" << m_iPos;
update();
}
}
Expand Down Expand Up @@ -247,14 +247,14 @@ void WOverview::mouseMoveEvent(QMouseEvent* e) {
void WOverview::mouseReleaseEvent(QMouseEvent* e) {
mouseMoveEvent(e);

float fValue = positionToValue(m_iPos);
double dValue = positionToValue(m_iPos);

//qDebug() << "WOverview::mouseReleaseEvent" << e->pos() << m_iPos << ">>" << fValue;
//qDebug() << "WOverview::mouseReleaseEvent" << e->pos() << m_iPos << ">>" << dValue;

if (e->button() == Qt::RightButton) {
emit(valueChangedRightUp(fValue));
emit(valueChangedRightUp(dValue));
} else {
emit(valueChangedLeftUp(fValue));
emit(valueChangedLeftUp(dValue));
}
m_bDrag = false;
}
Expand Down Expand Up @@ -466,9 +466,21 @@ void WOverview::paintText(const QString &text, QPainter *painter) {
}

void WOverview::resizeEvent(QResizeEvent *) {
//Those coeficient map position from [0;width-1] to value [14;114]
m_a = (float)((width()-1))/( 114.f - 14.f);
m_b = 14.f * m_a;
// Play-position potmeters range from -0.14 to 1.14. This is to give VC and
// MIDI control access to the pre-roll area.
// TODO(rryan): get these limits from the CO itself.
const double kMaxPlayposRange = 1.14;
const double kMinPlayposRange = -0.14;

// Values of zero and one in normalized space.
const double zero = (0.0 - kMinPlayposRange) / (kMaxPlayposRange - kMinPlayposRange);
const double one = (1.0 - kMinPlayposRange) / (kMaxPlayposRange - kMinPlayposRange);

// These coeficients convert between widget space and normalized value
// space.
m_a = (width() - 1) / (one - zero);
m_b = zero * m_a;

m_waveformImageScaled = QImage();
m_diffGain = 0;
}
Expand Down
10 changes: 5 additions & 5 deletions src/widget/woverview.h
Original file line number Diff line number Diff line change
Expand Up @@ -84,11 +84,11 @@ class WOverview : public WWidget {
// Append the waveform overview pixmap according to available data in waveform
virtual bool drawNextPixmapPart() = 0;
void paintText(const QString &text, QPainter *painter);
inline int valueToPosition(float value) const {
return static_cast<int>(m_a * value - m_b + 0.5);
inline int valueToPosition(double value) const {
return static_cast<int>(m_a * value - m_b);
}
inline double positionToValue(int position) const {
return (static_cast<float>(position) + m_b) / m_a;
return (static_cast<double>(position) + m_b) / m_a;
}

const QString m_group;
Expand Down Expand Up @@ -116,8 +116,8 @@ class WOverview : public WWidget {
std::vector<WaveformMarkRange> m_markRanges;

// Coefficient value-position linear transposition
float m_a;
float m_b;
double m_a;
double m_b;

double m_dAnalyserProgress;
bool m_bAnalyserFinalizing;
Expand Down
30 changes: 19 additions & 11 deletions src/widget/wslidercomposed.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -101,16 +101,19 @@ void WSliderComposed::mouseMoveEvent(QMouseEvent * e) {

int sliderLength = m_bHorizontal ? width() : height();

// Clamp to the range [0, sliderLength - m_iHandleLength].
if (m_iPos > (sliderLength - m_iHandleLength)) {
m_iPos = sliderLength - m_iHandleLength;
} else if (m_iPos < 0) {
m_iPos = 0;
}

// value ranges from 0 to 127
m_value = (double)m_iPos * (127. / (double)(sliderLength - m_iHandleLength));
// Divide by (sliderLength - m_iHandleLength) to produce a normalized
// value in the range of [0.0, 1.0]. 1.0
m_value = static_cast<double>(m_iPos) /
static_cast<double>(sliderLength - m_iHandleLength);
if (!m_bHorizontal) {
m_value = 127. - m_value;
m_value = 1.0 - m_value;
}

// Emit valueChanged signal
Expand All @@ -128,9 +131,14 @@ void WSliderComposed::mouseMoveEvent(QMouseEvent * e) {
}

void WSliderComposed::wheelEvent(QWheelEvent *e) {
double wheelDirection = ((QWheelEvent *)e)->delta() / 120.;
double newValue = getValue() + (wheelDirection);
this->updateValue(newValue);
// For legacy (MIDI) reasons this is tuned to 127.
double wheelDirection = ((QWheelEvent *)e)->delta() / (120.0 * 127.0);
double newValue = getValue() + wheelDirection;

// Clamp to [0.0, 1.0]
newValue = math_max(0.0, math_min(1.0, newValue));

updateValue(newValue);

e->accept();

Expand Down Expand Up @@ -192,18 +200,18 @@ void WSliderComposed::paintEvent(QPaintEvent *) {
}
}

void WSliderComposed::setValue(double fValue) {
if (!m_bDrag && m_value != fValue) {
void WSliderComposed::setValue(double dValue) {
if (!m_bDrag && m_value != dValue) {
// Set value without emitting a valueChanged signal
// and force display update
m_value = fValue;
m_value = dValue;

// Calculate handle position
if (!m_bHorizontal) {
fValue = 127-fValue;
dValue = 1.0 - dValue;
}
int sliderLength = m_bHorizontal ? width() : height();
m_iPos = (int)((fValue / 127.) * (double)(sliderLength - m_iHandleLength));
m_iPos = static_cast<int>(dValue * (sliderLength - m_iHandleLength));

if (m_iPos > (sliderLength - m_iHandleLength)) {
m_iPos = sliderLength - m_iHandleLength;
Expand Down
Loading

0 comments on commit bc4188a

Please sign in to comment.