MuseScore/libmscore/changeMap.h
2020-05-29 21:15:25 +02:00

113 lines
3.2 KiB
C++

//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2009-2019 Werner Schweer
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 2
// as published by the Free Software Foundation and appearing in
// the file LICENCE.GPL
//=============================================================================
#ifndef __CHANGEMAP_H__
#define __CHANGEMAP_H__
#include <QMultiMap>
#include "fraction.h"
/**
\file
Definition of class ChangeMap.
*/
namespace Ms {
enum class ChangeMethod : signed char {
NORMAL,
EXPONENTIAL,
EASE_IN,
EASE_OUT,
EASE_IN_OUT // and shake it all about
};
enum class ChangeDirection : signed char {
INCREASING,
DECREASING
};
//---------------------------------------------------------
/// ChangeEvent
/// item in ChangeMap
//---------------------------------------------------------
enum class ChangeEventType : char {
FIX, RAMP, INVALID
};
class ChangeEvent
{
int value { 0 };
ChangeEventType type { ChangeEventType::INVALID };
Fraction length;
ChangeMethod method { ChangeMethod::NORMAL };
ChangeDirection direction { ChangeDirection::INCREASING };
int cachedStartVal { -1 };
int cachedEndVal { -1 };
public:
ChangeEvent() {}
ChangeEvent(int vel)
: value(vel), type(ChangeEventType::FIX) {}
ChangeEvent(Fraction s, Fraction e, int diff, ChangeMethod m, ChangeDirection d)
: value(diff), type(ChangeEventType::RAMP), length(e - s), method(m), direction(d) {}
bool operator==(const ChangeEvent& event) const;
bool operator!=(const ChangeEvent& event) const { return !(operator==(event)); }
friend class ChangeMap;
};
//---------------------------------------------------------
// ChangeMap
/// List of changes in a value.
//---------------------------------------------------------
typedef std::vector<std::pair<Fraction, Fraction> > EndPointsVector;
class ChangeMap : public QMultiMap<Fraction, ChangeEvent>
{
bool cleanedUp { false };
static const int DEFAULT_VALUE { 80 }; // TODO
struct ChangeMethodItem {
ChangeMethod method;
const char* name;
};
static bool compareRampEvents(ChangeEvent& a, ChangeEvent& b) { return a.length > b.length; }
void cleanupStage0();
void cleanupStage1();
void cleanupStage2(std::vector<bool>& startsInRamp, EndPointsVector& endPoints);
void cleanupStage3();
public:
ChangeMap() {}
int val(Fraction tick);
std::vector<std::pair<Fraction, Fraction> > changesInRange(Fraction stick, Fraction etick);
void addFixed(Fraction tick, int value);
void addRamp(Fraction stick, Fraction etick, int change, ChangeMethod method, ChangeDirection direction);
void cleanup();
void dump();
static int interpolate(Fraction& eventTick, ChangeEvent& event, Fraction& tick);
static QString changeMethodToName(ChangeMethod method);
static ChangeMethod nameToChangeMethod(QString name);
static const std::vector<ChangeMethodItem> changeMethodTable;
};
} // namespace Ms
#endif