1f9ccfcdce
git-subtree-dir: libmscore git-subtree-mainline:412ca45401
git-subtree-split:6047361bd0
235 lines
7.4 KiB
C++
235 lines
7.4 KiB
C++
//=============================================================================
|
|
// MuseScore
|
|
// Music Composition & Notation
|
|
// $Id: hairpin.cpp 5305 2012-02-09 12:09:20Z wschweer $
|
|
//
|
|
// 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 "hairpin.h"
|
|
#include "style.h"
|
|
#include "xml.h"
|
|
#include "utils.h"
|
|
#include "score.h"
|
|
#include "measure.h"
|
|
#include "segment.h"
|
|
#include "system.h"
|
|
#include "undo.h"
|
|
#include "staff.h"
|
|
#include "mscore.h"
|
|
|
|
//---------------------------------------------------------
|
|
// propertyList
|
|
//---------------------------------------------------------
|
|
|
|
static int defaultSubtype = 0;
|
|
|
|
Property<Hairpin> Hairpin::propertyList[] = {
|
|
{ P_SUBTYPE, &Hairpin::pSubtype, &defaultSubtype },
|
|
};
|
|
static const int PROPERTIES = sizeof(Hairpin::propertyList)/sizeof(*Hairpin::propertyList);
|
|
|
|
//---------------------------------------------------------
|
|
// layout
|
|
//---------------------------------------------------------
|
|
|
|
void HairpinSegment::layout()
|
|
{
|
|
QTransform t;
|
|
qreal _spatium = spatium();
|
|
qreal h1 = score()->styleS(ST_hairpinHeight).val() * _spatium * .5;
|
|
qreal h2 = score()->styleS(ST_hairpinContHeight).val() * _spatium * .5;
|
|
|
|
qreal len;
|
|
qreal x = pos2().x();
|
|
if (x < _spatium) // minimum size of hairpin
|
|
x = _spatium;
|
|
qreal y = pos2().y();
|
|
len = sqrt(x * x + y * y);
|
|
t.rotateRadians(asin(y/len));
|
|
|
|
if (hairpin()->subtype() == 0) {
|
|
// crescendo
|
|
switch (subtype()) {
|
|
case SEGMENT_SINGLE:
|
|
case SEGMENT_BEGIN:
|
|
l1 = QLineF(.0, .0, len, h1);
|
|
l2 = QLineF(.0, .0, len, - h1);
|
|
break;
|
|
case SEGMENT_MIDDLE:
|
|
case SEGMENT_END:
|
|
l1 = QLineF(.0, h2, len, h1);
|
|
l2 = QLineF(.0, -h2, len, - h1);
|
|
break;
|
|
}
|
|
}
|
|
else {
|
|
// decrescendo
|
|
switch(subtype()) {
|
|
case SEGMENT_SINGLE:
|
|
case SEGMENT_END:
|
|
l1 = QLineF(.0, h1, len, 0.0);
|
|
l2 = QLineF(.0, -h1, len, 0.0);
|
|
break;
|
|
case SEGMENT_BEGIN:
|
|
case SEGMENT_MIDDLE:
|
|
l1 = QLineF(.0, h1, len, + h2);
|
|
l2 = QLineF(.0, -h1, len, - h2);
|
|
break;
|
|
}
|
|
}
|
|
l1 = t.map(l1);
|
|
l2 = t.map(l2);
|
|
|
|
QRectF r = QRectF(l1.p1(), l1.p2()).normalized() | QRectF(l2.p1(), l2.p2()).normalized();
|
|
qreal w = point(score()->styleS(ST_hairpinWidth));
|
|
setbbox(r.adjusted(-w*.5, -w*.5, w, w));
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// draw
|
|
//---------------------------------------------------------
|
|
|
|
void HairpinSegment::draw(QPainter* painter) const
|
|
{
|
|
QPen pen(curColor(), point(score()->styleS(ST_hairpinWidth)));
|
|
painter->setPen(pen);
|
|
painter->drawLine(QLineF(l1.x1(), l1.y1(), l1.x2(), l1.y2()));
|
|
painter->drawLine(QLineF(l2.x1(), l2.y1(), l2.x2(), l2.y2()));
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// Hairpin
|
|
//---------------------------------------------------------
|
|
|
|
Hairpin::Hairpin(Score* s)
|
|
: SLine(s)
|
|
{
|
|
_subtype = 0;
|
|
_veloChange = 10;
|
|
_dynType = DYNAMIC_PART;
|
|
setYoff(s->styleS(ST_hairpinY).val());
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// layout
|
|
// compute segments from tick() to _tick2
|
|
//---------------------------------------------------------
|
|
|
|
void Hairpin::layout()
|
|
{
|
|
setPos(0.0, 0.0);
|
|
SLine::layout();
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// createLineSegment
|
|
//---------------------------------------------------------
|
|
|
|
LineSegment* Hairpin::createLineSegment()
|
|
{
|
|
return new HairpinSegment(score());
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// write
|
|
//---------------------------------------------------------
|
|
|
|
void Hairpin::write(Xml& xml) const
|
|
{
|
|
xml.stag(QString("%1 id=\"%2\"").arg(name()).arg(id()));
|
|
xml.tag("subtype", _subtype);
|
|
xml.tag("veloChange", _veloChange);
|
|
if (_dynType != DYNAMIC_PART)
|
|
xml.tag("dynType", _dynType);
|
|
SLine::writeProperties(xml);
|
|
xml.etag();
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// read
|
|
//---------------------------------------------------------
|
|
|
|
void Hairpin::read(const QDomElement& de)
|
|
{
|
|
foreach(SpannerSegment* seg, spannerSegments())
|
|
delete seg;
|
|
spannerSegments().clear();
|
|
setId(de.attribute("id", "-1").toInt());
|
|
for (QDomElement e = de.firstChildElement(); !e.isNull(); e = e.nextSiblingElement()) {
|
|
const QString& tag(e.tagName());
|
|
const QString& val(e.text());
|
|
if (tag == "subtype")
|
|
_subtype = val.toInt();
|
|
else if (tag == "veloChange")
|
|
_veloChange = val.toInt();
|
|
else if (e.tagName() == "dynType")
|
|
_dynType = DynamicType(val.toInt());
|
|
else if (!SLine::readProperties(e))
|
|
domError(e);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// property
|
|
//---------------------------------------------------------
|
|
|
|
Property<Hairpin>* Hairpin::property(P_ID id) const
|
|
{
|
|
for (int i = 0; i < PROPERTIES; ++i) {
|
|
if (propertyList[i].id == id)
|
|
return &propertyList[i];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// getProperty
|
|
//---------------------------------------------------------
|
|
|
|
QVariant Hairpin::getProperty(P_ID propertyId) const
|
|
{
|
|
Property<Hairpin>* p = property(propertyId);
|
|
if (p)
|
|
return getVariant(propertyId, ((*(Hairpin*)this).*(p->data))());
|
|
return Element::getProperty(propertyId);
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// setProperty
|
|
//---------------------------------------------------------
|
|
|
|
bool Hairpin::setProperty(P_ID propertyId, const QVariant& v)
|
|
{
|
|
Property<Hairpin>* p = property(propertyId);
|
|
if (p) {
|
|
setVariant(propertyId, ((*this).*(p->data))(), v);
|
|
setGenerated(false);
|
|
return true;
|
|
}
|
|
return Element::setProperty(propertyId, v);
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// setProperty
|
|
//---------------------------------------------------------
|
|
|
|
bool Hairpin::setProperty(const QString& name, const QDomElement& e)
|
|
{
|
|
for (int i = 0; i < PROPERTIES; ++i) {
|
|
P_ID id = propertyList[i].id;
|
|
if (propertyName(id) == name) {
|
|
QVariant v = ::getProperty(id, e);
|
|
setVariant(id, ((*this).*(propertyList[i].data))(), v);
|
|
setGenerated(false);
|
|
return true;
|
|
}
|
|
}
|
|
return Element::setProperty(name, e);
|
|
}
|
|
|