MuseScore/libmscore/dynamic.cpp

402 lines
15 KiB
C++
Raw Normal View History

2012-05-26 14:26:10 +02:00
//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2002-2011 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
//=============================================================================
#include "dynamic.h"
#include "xml.h"
#include "score.h"
#include "measure.h"
#include "system.h"
#include "segment.h"
#include "utils.h"
#include "style.h"
#include "mscore.h"
2013-08-12 14:57:17 +02:00
#include "chord.h"
2013-10-22 12:05:31 +02:00
#include "undo.h"
#include "sym.h"
2012-05-26 14:26:10 +02:00
2013-05-13 18:49:17 +02:00
namespace Ms {
//-----------------------------------------------------------------------------
// Dyn
// see: http://en.wikipedia.org/wiki/File:Dynamic's_Note_Velocity.svg
//-----------------------------------------------------------------------------
struct Dyn {
int velocity; ///< associated midi velocity (0-127, -1 = none)
bool accent; ///< if true add velocity to current chord velocity
const char* tag; // name of dynamics, eg. "fff"
const char* text; // utf8 text of dynamic
};
#if 1
// variant with ligatures, works for both emmentaler and bravura:
static Dyn dynList[] = {
2012-05-26 14:26:10 +02:00
// dynamic:
{ -1, true, "other-dynamics", "" },
{ 1, false, "pppppp", "<sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym>" },
2014-02-21 10:12:30 +01:00
{ 5, false, "ppppp", "<sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym>" },
{ 10, false, "pppp", "<sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym>" },
{ 16, false, "ppp", "<sym>dynamicPiano</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym>" },
{ 33, false, "pp", "<sym>dynamicPiano</sym><sym>dynamicPiano</sym>" },
{ 49, false, "p", "<sym>dynamicPiano</sym>" },
{ 64, false, "mp", "<sym>dynamicMezzo</sym><sym>dynamicPiano</sym>" },
{ 80, false, "mf", "<sym>dynamicMezzo</sym><sym>dynamicForte</sym>" },
{ 96, false, "f", "<sym>dynamicForte</sym>" },
{ 112, false, "ff", "<sym>dynamicForte</sym><sym>dynamicForte</sym>" },
{ 126, false, "fff", "<sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym>" },
{ 127, false, "ffff", "<sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym>" },
{ 127, false, "fffff", "<sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym>" },
{ 127, false, "ffffff", "<sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicForte</sym>" },
2012-05-26 14:26:10 +02:00
// accents:
{ 0, true, "fp", "<sym>dynamicForte</sym><sym>dynamicPiano</sym>"},
{ 0, true, "sf", "<sym>dynamicSforzando</sym><sym>dynamicForte</sym>"},
{ 0, true, "sfz", "<sym>dynamicSforzando</sym><sym>dynamicForte</sym><sym>dynamicZ</sym>"},
{ 0, true, "sff", "<sym>dynamicSforzando</sym><sym>dynamicForte</sym><sym>dynamicForte</sym>"},
{ 0, true, "sffz", "<sym>dynamicSforzando</sym><sym>dynamicForte</sym><sym>dynamicForte</sym><sym>dynamicZ</sym>"},
{ 0, true, "sfp", "<sym>dynamicSforzando</sym><sym>dynamicForte</sym>"},
{ 0, true, "sfpp", "<sym>dynamicSforzando</sym><sym>dynamicForte</sym><sym>dynamicPiano</sym><sym>dynamicPiano</sym>"},
{ 0, true, "rfz", "<sym>dynamicRinforzando</sym><sym>dynamicForte</sym><sym>dynamicZ</sym>"},
{ 0, true, "rf", "<sym>dynamicRinforzando</sym><sym>dynamicForte</sym>"},
{ 0, true, "fz", "<sym>dynamicForte</sym><sym>dynamicZ</sym>"},
{ 0, true, "m", "<sym>dynamicMezzo</sym>"},
{ 0, true, "r", "<sym>dynamicRinforzando</sym>"},
{ 0, true, "s", "<sym>dynamicSforzando</sym>"},
{ 0, true, "z", "<sym>dynamicZ</sym>"},
{ 0, true, "n", "<sym>dynamicNiente</sym>" }
2012-05-26 14:26:10 +02:00
};
2013-08-12 13:15:46 +02:00
// variant with precomposed symbols, available only in bravura:
2013-08-12 13:15:46 +02:00
#else
2013-08-12 13:15:46 +02:00
static Dyn dynList[] = {
// dynamic:
{ -1, true, "other-dynamics", "" },
{ 1, false, "pppppp", "<sym>dynamicPPPPPP</sym>" },
{ 5, false, "ppppp", "<sym>dynamicPPPPP</sym>" },
{ 10, false, "pppp", "<sym>dynamicPPPP</sym>" },
{ 16, false, "ppp", "<sym>dynamicPPP</sym>" },
{ 33, false, "pp", "<sym>dynamicPP</sym>" },
{ 49, false, "p", "<sym>dynamicPiano</sym>" },
{ 64, false, "mp", "<sym>dynamicMP</sym>" },
{ 80, false, "mf", "<sym>dynamicMF</sym>" },
{ 96, false, "f", "<sym>dynamicForte</sym>" },
{ 112, false, "ff", "<sym>dynamicFF</sym>" },
{ 126, false, "fff", "<sym>dynamicFFF</sym>" },
{ 127, false, "ffff", "<sym>dynamicFFFF</sym>" },
{ 127, false, "fffff", "<sym>dynamicFFFFF</sym>" },
{ 127, false, "ffffff", "<sym>dynamicFFFFFF</sym>" },
2013-08-12 13:15:46 +02:00
// accents:
{ 0, true, "fp", "<sym>dynamicFortePiano</sym>" },
{ 0, true, "sf", "<sym>dynamicSforzando1</sym>" },
{ 0, true, "sfz", "<sym>dynamicSforzato</sym>" },
{ 0, true, "sff", "<sym>dynamicSforzando</sym><sym>dynamicFF</sym>" },
{ 0, true, "sffz", "<sym>dynamicSforzatoFF</sym>" },
{ 0, true, "sfp", "<sym>dynamicSforzandoPiano</sym>" },
{ 0, true, "sfpp", "<sym>dynamicSforzandoPianissimo</sym>" },
{ 0, true, "rfz", "<sym>dynamicRinforzando2</sym>" },
{ 0, true, "rf", "<sym>dynamicRinforzando1</sym>" },
{ 0, true, "fz", "<sym>dynamicForzando</sym>" },
{ 0, true, "m", "<sym>dynamicMezzo</sym>" },
{ 0, true, "r", "<sym>dynamicRinforzando</sym>" },
{ 0, true, "s", "<sym>dynamicSforzando</sym>" },
{ 0, true, "z", "<sym>dynamicZ</sym>" },
{ 0, true, "n", "<sym>dynamicNiente</sym>" }
2013-08-12 13:15:46 +02:00
};
#endif
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
// Dynamic
//---------------------------------------------------------
Dynamic::Dynamic(Score* s)
: Text(s)
{
2014-03-24 14:01:10 +01:00
setFlags(ELEMENT_MOVABLE | ELEMENT_SELECTABLE | ELEMENT_ON_STAFF);
2012-05-26 14:26:10 +02:00
_velocity = -1;
_dynRange = DYNAMIC_PART;
setTextStyleType(TEXT_STYLE_DYNAMICS);
_dynamicType = DYNAMIC_OTHER;
2012-05-26 14:26:10 +02:00
}
Dynamic::Dynamic(const Dynamic& d)
: Text(d)
{
2013-10-22 12:05:31 +02:00
_dynamicType = d._dynamicType;
_velocity = d._velocity;
_dynRange = d._dynRange;
2012-05-26 14:26:10 +02:00
}
//---------------------------------------------------------
// setVelocity
//---------------------------------------------------------
void Dynamic::setVelocity(int v)
{
_velocity = v;
}
//---------------------------------------------------------
// velocity
//---------------------------------------------------------
int Dynamic::velocity() const
{
return _velocity <= 0 ? dynList[dynamicType()].velocity : _velocity;
2012-05-26 14:26:10 +02:00
}
//---------------------------------------------------------
// write
//---------------------------------------------------------
void Dynamic::write(Xml& xml) const
{
xml.stag("Dynamic");
xml.tag("subtype", dynamicTypeName());
writeProperty(xml, P_VELOCITY);
writeProperty(xml, P_DYNAMIC_RANGE);
Text::writeProperties(xml, dynamicType() == 0);
2012-05-26 14:26:10 +02:00
xml.etag();
}
//---------------------------------------------------------
// read
//---------------------------------------------------------
2013-01-11 18:10:18 +01:00
void Dynamic::read(XmlReader& e)
2012-05-26 14:26:10 +02:00
{
2013-01-11 18:10:18 +01:00
while (e.readNextStartElement()) {
const QStringRef& tag = e.name();
2013-02-14 13:50:59 +01:00
if (tag == "subtype") {
setDynamicType(e.readElementText());
2013-02-14 13:50:59 +01:00
}
else if (tag == "velocity")
2013-01-11 18:10:18 +01:00
_velocity = e.readInt();
else if (tag == "dynType")
2013-01-11 18:10:18 +01:00
_dynRange = DynamicRange(e.readInt());
2012-05-26 14:26:10 +02:00
else if (!Text::readProperties(e))
2013-01-11 18:10:18 +01:00
e.unknown();
2012-05-26 14:26:10 +02:00
}
2014-03-14 18:34:07 +01:00
if (textStyleType() == TEXT_STYLE_DEFAULT)
setTextStyleType(TEXT_STYLE_DYNAMICS);
2012-10-24 13:15:46 +02:00
}
//---------------------------------------------------------
// layout
//---------------------------------------------------------
void Dynamic::layout()
{
if (!readPos().isNull()) {
if (score()->mscVersion() < 118) {
setReadPos(QPointF());
// hack: 1.2 boundingBoxes are a bit wider which results
// in symbols moved right
setUserXoffset(userOff().x() - spatium() * .6);
}
}
2014-03-14 19:10:48 +01:00
setPos(textStyle().offset(spatium()));
Text::layout1();
2013-08-12 14:57:17 +02:00
Segment* s = segment();
2013-10-22 12:05:31 +02:00
if (!s)
return;
2013-08-12 14:57:17 +02:00
for (int voice = 0; voice < VOICES; ++voice) {
int t = (track() & ~0x3) + voice;
Chord* c = static_cast<Chord*>(s->element(t));
if (!c)
continue;
if (c->type() == CHORD) {
2013-08-12 16:41:15 +02:00
qreal noteHeadWidth = score()->noteHeadWidth() * c->mag();
2013-08-12 14:57:17 +02:00
if (c->stem() && !c->up()) // stem down
2013-08-12 16:41:15 +02:00
rxpos() += noteHeadWidth * .25; // center on stem + optical correction
2013-08-12 14:57:17 +02:00
else
2013-08-12 16:41:15 +02:00
rxpos() += noteHeadWidth * .5; // center on note head
2013-08-12 14:57:17 +02:00
}
else
rxpos() += c->width() * .5;
break;
}
2014-03-14 19:10:48 +01:00
adjustReadPos();
2012-05-26 14:26:10 +02:00
}
//---------------------------------------------------------
// setDynamicType
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
void Dynamic::setDynamicType(const QString& tag)
2012-05-26 14:26:10 +02:00
{
int n = sizeof(dynList)/sizeof(*dynList);
2013-02-14 13:50:59 +01:00
for (int i = 0; i < n; ++i) {
2014-04-14 13:16:46 +02:00
if (dynList[i].tag == tag || dynList[i].text == tag) {
setDynamicType(DynamicType(i));
setText(QString::fromUtf8(dynList[i].text));
2012-05-26 14:26:10 +02:00
return;
}
}
2013-08-12 13:15:46 +02:00
qDebug("setDynamicType: other <%s>", qPrintable(tag));
setDynamicType(DYNAMIC_OTHER);
2012-05-26 14:26:10 +02:00
setText(tag);
}
//---------------------------------------------------------
// dynamicTypeName
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
QString Dynamic::dynamicTypeName() const
2012-05-26 14:26:10 +02:00
{
return dynList[dynamicType()].tag;
2012-05-26 14:26:10 +02:00
}
//---------------------------------------------------------
// startEdit
//---------------------------------------------------------
void Dynamic::startEdit(MuseScoreView* v, const QPointF& p)
{
Text::startEdit(v, p);
}
2013-10-22 12:05:31 +02:00
//---------------------------------------------------------
// endEdit
//---------------------------------------------------------
void Dynamic::endEdit()
{
Text::endEdit();
2014-03-14 11:30:19 +01:00
if (text() != QString::fromUtf8(dynList[_dynamicType].text))
_dynamicType = DYNAMIC_OTHER;
}
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
2012-11-19 10:08:15 +01:00
// reset
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
2012-11-19 10:08:15 +01:00
void Dynamic::reset()
2012-05-26 14:26:10 +02:00
{
// setDynamicType(getText());
2012-11-19 10:08:15 +01:00
Text::reset();
2012-05-26 14:26:10 +02:00
}
//---------------------------------------------------------
// dragAnchor
//---------------------------------------------------------
QLineF Dynamic::dragAnchor() const
{
qreal xp = 0.0;
for (Element* e = parent(); e; e = e->parent())
xp += e->x();
2013-08-30 12:46:15 +02:00
qreal yp = measure()->system()->staffYpage(staffIdx());
2012-05-26 14:26:10 +02:00
QPointF p(xp, yp);
2014-03-24 13:23:54 +01:00
2012-05-26 14:26:10 +02:00
return QLineF(p, canvasPos());
}
2013-10-22 12:05:31 +02:00
//---------------------------------------------------------
// drag
//---------------------------------------------------------
QRectF Dynamic::drag(EditData* ed)
{
QRectF f = Element::drag(ed);
//
// move anchor
//
Qt::KeyboardModifiers km = qApp->keyboardModifiers();
if (km != (Qt::ShiftModifier | Qt::ControlModifier)) {
int si;
Segment* seg = 0;
_score->pos2measure(ed->pos, &si, 0, &seg, 0);
if (seg && (seg != segment() || staffIdx() != si)) {
2014-03-24 13:23:54 +01:00
QPointF pos1(canvasPos());
2013-10-22 12:05:31 +02:00
score()->undo(new ChangeParent(this, seg, si));
setUserOff(QPointF());
layout();
2014-03-24 13:23:54 +01:00
QPointF pos2(canvasPos());
2013-10-22 12:05:31 +02:00
setUserOff(pos1 - pos2);
ed->startMove = pos2;
}
}
return f;
}
//---------------------------------------------------------
// undoSetDynRange
//---------------------------------------------------------
void Dynamic::undoSetDynRange(DynamicRange v)
{
score()->undoChangeProperty(this, P_DYNAMIC_RANGE, v);
}
//---------------------------------------------------------
// getProperty
//---------------------------------------------------------
QVariant Dynamic::getProperty(P_ID propertyId) const
{
switch(propertyId) {
case P_DYNAMIC_RANGE: return int(_dynRange);
case P_VELOCITY: return velocity();
case P_SUBTYPE: return _dynamicType;
default:
return Text::getProperty(propertyId);
}
}
//---------------------------------------------------------
// setProperty
//---------------------------------------------------------
bool Dynamic::setProperty(P_ID propertyId, const QVariant& v)
{
switch (propertyId) {
case P_DYNAMIC_RANGE:
_dynRange = DynamicRange(v.toInt());
break;
case P_VELOCITY:
_velocity = v.toInt();
break;
case P_SUBTYPE:
_dynamicType = DynamicType(v.toInt());
break;
default:
if (!Text::setProperty(propertyId, v))
return false;
break;
}
score()->setLayoutAll(true);
return true;
}
//---------------------------------------------------------
// propertyDefault
//---------------------------------------------------------
QVariant Dynamic::propertyDefault(P_ID id) const
{
switch(id) {
2014-03-14 18:34:07 +01:00
case P_TEXT_STYLE_TYPE: return TEXT_STYLE_DYNAMICS;
case P_DYNAMIC_RANGE: return DYNAMIC_PART;
case P_VELOCITY: return -1;
default: return Text::propertyDefault(id);
}
}
2013-05-13 18:49:17 +02:00
}