MuseScore/libmscore/marker.cpp

340 lines
10 KiB
C++
Raw Normal View History

2013-02-25 14:56:34 +01:00
//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2002-2013 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 "marker.h"
#include "score.h"
2013-11-11 16:53:03 +01:00
#include "sym.h"
2014-04-09 16:09:21 +02:00
#include "xml.h"
#include "measure.h"
2013-02-25 14:56:34 +01:00
2013-05-13 18:49:17 +02:00
namespace Ms {
//must be in sync with Marker::Type enum
const MarkerTypeItem markerTypeTable[] = {
{ Marker::Type::SEGNO , QT_TRANSLATE_NOOP("markerType", "Segno") },
2014-10-15 20:43:06 +02:00
{ Marker::Type::VARSEGNO, QT_TRANSLATE_NOOP("markerType", "Segno variation")},
{ Marker::Type::CODA , QT_TRANSLATE_NOOP("markerType", "Coda") },
{ Marker::Type::VARCODA , QT_TRANSLATE_NOOP("markerType", "Varied coda") },
{ Marker::Type::CODETTA , QT_TRANSLATE_NOOP("markerType", "Codetta") },
{ Marker::Type::FINE , QT_TRANSLATE_NOOP("markerType", "Fine") },
{ Marker::Type::TOCODA , QT_TRANSLATE_NOOP("markerType", "To Coda") },
{ Marker::Type::USER , QT_TRANSLATE_NOOP("markerType", "Custom") }
};
int markerTypeTableSize()
{
return sizeof(markerTypeTable)/sizeof(MarkerTypeItem) - 1; //-1 for the user defined
}
2013-02-25 14:56:34 +01:00
//---------------------------------------------------------
// Marker
//---------------------------------------------------------
Marker::Marker(Score* s)
: Text(s)
{
_markerType = Type::FINE;
setFlags(ElementFlag::MOVABLE | ElementFlag::SELECTABLE | ElementFlag::ON_STAFF);
setTextStyleType(TextStyleType::REPEAT_LEFT);
setLayoutToParentWidth(true);
2013-02-25 14:56:34 +01:00
}
//---------------------------------------------------------
// setMarkerType
//---------------------------------------------------------
void Marker::setMarkerType(Type t)
2013-02-25 14:56:34 +01:00
{
_markerType = t;
2014-08-06 15:28:12 +02:00
const char* txt = 0;
2014-01-16 12:33:23 +01:00
switch (t) {
case Type::SEGNO:
2014-08-06 15:28:12 +02:00
txt = "<sym>segno</sym>";
2013-02-25 14:56:34 +01:00
setLabel("segno");
break;
case Type::VARSEGNO:
2014-08-06 15:28:12 +02:00
txt = "<sym>segnoSerpent1</sym>";
2013-02-25 14:56:34 +01:00
setLabel("varsegno");
break;
case Type::CODA:
2014-08-06 15:28:12 +02:00
txt = "<sym>coda</sym>";
2013-02-25 14:56:34 +01:00
setLabel("codab");
break;
case Type::VARCODA:
2014-08-06 15:28:12 +02:00
txt = "<sym>codaSquare</sym>";
2013-02-25 14:56:34 +01:00
setLabel("varcoda");
break;
case Type::CODETTA:
2014-08-06 15:28:12 +02:00
txt = "<sym>coda</sym><sym>coda</sym>";
2013-02-25 14:56:34 +01:00
setLabel("codetta");
break;
case Type::FINE:
2014-08-06 15:28:12 +02:00
txt = "Fine";
setTextStyleType(TextStyleType::REPEAT_RIGHT);
2013-02-25 14:56:34 +01:00
setLabel("fine");
break;
case Type::TOCODA:
2014-08-06 15:28:12 +02:00
txt = "To Coda";
setTextStyleType(TextStyleType::REPEAT_RIGHT);
2013-02-25 14:56:34 +01:00
setLabel("coda");
break;
case Type::USER:
2013-02-25 14:56:34 +01:00
break;
default:
qDebug("unknown marker type %hhd", t);
2013-02-25 14:56:34 +01:00
break;
}
2014-08-06 15:28:12 +02:00
if (isEmpty() && txt)
setText(txt);
2013-02-25 14:56:34 +01:00
}
QString Marker::markerTypeUserName()
{
2014-08-26 17:39:31 +02:00
return qApp->translate("markerType", markerTypeTable[static_cast<int>(_markerType)].name.toUtf8().constData());
}
2013-02-25 14:56:34 +01:00
//---------------------------------------------------------
// styleChanged
//---------------------------------------------------------
2014-01-16 12:33:23 +01:00
void Marker::styleChanged()
{
setMarkerType(_markerType);
}
2013-02-25 14:56:34 +01:00
//---------------------------------------------------------
// adjustReadPos
//---------------------------------------------------------
void Marker::adjustReadPos()
{
if (!readPos().isNull()) {
2013-02-25 14:56:34 +01:00
QPointF uo;
/*
2013-02-25 14:56:34 +01:00
if (score()->mscVersion() <= 114) {
// rebase from Measure to Segment
uo = userOff();
uo.rx() -= segment()->pos().x();
// 1.2 is always HCENTER aligned
if ((textStyle().align() & AlignmentFlags::HMASK) == 0) // AlignmentFlags::LEFT
2013-02-25 14:56:34 +01:00
uo.rx() -= bbox().width() * .5;
}
else
*/
2013-02-25 14:56:34 +01:00
uo = readPos() - ipos();
setUserOff(uo);
setReadPos(QPointF());
}
}
//---------------------------------------------------------
// markerType
//---------------------------------------------------------
Marker::Type Marker::markerType(const QString& s) const
2013-02-25 14:56:34 +01:00
{
if (s == "segno")
return Type::SEGNO;
2013-02-25 14:56:34 +01:00
else if (s == "varsegno")
return Type::VARSEGNO;
2013-02-25 14:56:34 +01:00
else if (s == "codab")
return Type::CODA;
2013-02-25 14:56:34 +01:00
else if (s == "varcoda")
return Type::VARCODA;
2013-02-25 14:56:34 +01:00
else if (s == "codetta")
return Type::CODETTA;
2013-02-25 14:56:34 +01:00
else if (s == "fine")
return Type::FINE;
2013-02-25 14:56:34 +01:00
else if (s == "coda")
return Type::TOCODA;
2013-02-25 14:56:34 +01:00
else
return Type::USER;
2013-02-25 14:56:34 +01:00
}
//---------------------------------------------------------
// layout
//---------------------------------------------------------
void Marker::layout()
{
setPos(textStyle().offset(spatium()));
Text::layout1();
// although normally laid out to parent (measure) width,
// force to center over barline if left-aligned
if (layoutToParentWidth() && !(textStyle().align() & (AlignmentFlags::RIGHT|AlignmentFlags::HCENTER)))
rxpos() -= width() * 0.5;
adjustReadPos();
}
2013-02-25 14:56:34 +01:00
//---------------------------------------------------------
// read
//---------------------------------------------------------
void Marker::read(XmlReader& e)
{
Type mt = Type::SEGNO;
2013-02-25 14:56:34 +01:00
while (e.readNextStartElement()) {
const QStringRef& tag(e.name());
if (tag == "label") {
QString s(e.readElementText());
setLabel(s);
mt = markerType(s);
}
else if (!Text::readProperties(e))
e.unknown();
}
// REPEAT is obsolete, but was previously used for both left and right aligned text
if (textStyleType() == TextStyleType::REPEAT)
setTextStyleType(TextStyleType::REPEAT_LEFT);
2014-01-16 12:33:23 +01:00
setMarkerType(mt);
2013-02-25 14:56:34 +01:00
}
//---------------------------------------------------------
// write
//---------------------------------------------------------
void Marker::write(Xml& xml) const
{
xml.stag(name());
Text::writeProperties(xml);
xml.tag("label", _label);
xml.etag();
}
//---------------------------------------------------------
// undoSetLabel
//---------------------------------------------------------
void Marker::undoSetLabel(const QString& s)
{
2014-05-26 18:18:01 +02:00
score()->undoChangeProperty(this, P_ID::LABEL, s);
2013-02-25 14:56:34 +01:00
}
//---------------------------------------------------------
// undoSetMarkerType
//---------------------------------------------------------
void Marker::undoSetMarkerType(Type t)
2013-02-25 14:56:34 +01:00
{
2014-05-26 18:18:01 +02:00
score()->undoChangeProperty(this, P_ID::MARKER_TYPE, int(t));
2013-02-25 14:56:34 +01:00
}
//---------------------------------------------------------
// getProperty
//---------------------------------------------------------
QVariant Marker::getProperty(P_ID propertyId) const
{
switch (propertyId) {
2014-05-26 18:18:01 +02:00
case P_ID::LABEL:
2013-02-25 14:56:34 +01:00
return label();
2014-05-26 18:18:01 +02:00
case P_ID::MARKER_TYPE:
2013-02-25 14:56:34 +01:00
return int(markerType());
default:
break;
}
return Text::getProperty(propertyId);
}
//---------------------------------------------------------
// setProperty
//---------------------------------------------------------
bool Marker::setProperty(P_ID propertyId, const QVariant& v)
{
switch (propertyId) {
2014-05-26 18:18:01 +02:00
case P_ID::LABEL:
2013-02-25 14:56:34 +01:00
setLabel(v.toString());
break;
2014-05-26 18:18:01 +02:00
case P_ID::MARKER_TYPE:
setMarkerType(Type(v.toInt()));
2013-02-25 14:56:34 +01:00
break;
default:
if (!Text::setProperty(propertyId, v))
return false;
break;
}
score()->setLayoutAll(true);
return true;
}
//---------------------------------------------------------
// propertyDefault
//---------------------------------------------------------
QVariant Marker::propertyDefault(P_ID propertyId) const
{
switch (propertyId) {
2014-05-26 18:18:01 +02:00
case P_ID::LABEL:
2013-02-25 14:56:34 +01:00
return QString();
2014-05-26 18:18:01 +02:00
case P_ID::MARKER_TYPE:
return int(Type::FINE);
2013-02-25 14:56:34 +01:00
default:
break;
}
return Text::propertyDefault(propertyId);
}
//---------------------------------------------------------
// nextElement
//---------------------------------------------------------
Element* Marker::nextElement()
{
Segment* seg;
if (markerType() == Marker::Type::FINE) {
seg = measure()->last();
return seg->firstElement(staffIdx());
}
Measure* prevMeasure = measure()->prevMeasureMM();
if (prevMeasure) {
seg = prevMeasure->last();
return seg->firstElement(staffIdx());
}
return Element::nextElement();
}
//---------------------------------------------------------
// prevElement
//---------------------------------------------------------
2013-02-25 14:56:34 +01:00
Element* Marker::prevElement()
{
//it's the same barline
return nextElement();
}
2013-02-25 14:56:34 +01:00
//---------------------------------------------------------
// accessibleInfo
//---------------------------------------------------------
QString Marker::accessibleInfo()
{
return QString("%1: %2").arg(Element::accessibleInfo()).arg(markerTypeUserName());
}
2013-05-13 18:49:17 +02:00
}