MuseScore/libmscore/layoutbreak.cpp
2013-05-15 15:31:29 +02:00

260 lines
7.9 KiB
C++

//=============================================================================
// 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 "layoutbreak.h"
#include "score.h"
#include "mscore.h"
namespace Ms {
//---------------------------------------------------------
// LayoutBreak
//---------------------------------------------------------
LayoutBreak::LayoutBreak(Score* score)
: Element(score)
{
_layoutBreakType = LayoutBreakType(propertyDefault(P_LAYOUT_BREAK).toInt());
_pause = score->styleD(ST_SectionPause);
_startWithLongNames = true;
_startWithMeasureOne = true;
lw = spatium() * 0.3;
setFlag(ELEMENT_HAS_TAG, true);
}
//---------------------------------------------------------
// write
//---------------------------------------------------------
void LayoutBreak::write(Xml& xml) const
{
xml.stag(name());
Element::writeProperties(xml);
writeProperty(xml, P_LAYOUT_BREAK);
writeProperty(xml, P_PAUSE);
if (!_startWithLongNames)
xml.tag("startWithLongNames", _startWithLongNames);
if (!_startWithMeasureOne)
xml.tag("startWithMeasureOne", _startWithMeasureOne);
xml.etag();
}
//---------------------------------------------------------
// read
//---------------------------------------------------------
void LayoutBreak::read(XmlReader& e)
{
while (e.readNextStartElement()) {
const QStringRef& tag(e.name());
if (tag == "subtype")
setProperty(P_LAYOUT_BREAK, Ms::getProperty(P_LAYOUT_BREAK, e));
else if (tag == "pause")
_pause = e.readDouble();
else if (tag == "startWithLongNames")
_startWithLongNames = e.readInt();
else if (tag == "startWithMeasureOne")
_startWithMeasureOne = e.readInt();
else if (!Element::readProperties(e))
e.unknown();
}
layout0();
}
//---------------------------------------------------------
// draw
//---------------------------------------------------------
void LayoutBreak::draw(QPainter* painter) const
{
if (score()->printing() || !score()->showUnprintable())
return;
QPainterPathStroker stroker;
stroker.setWidth(lw/2);
stroker.setJoinStyle(Qt::MiterJoin);
stroker.setCapStyle(Qt::SquareCap);
QVector<qreal> dashes ;
dashes.append(1);
dashes.append(3);
stroker.setDashPattern(dashes);
QPainterPath stroke = stroker.createStroke(path);
painter->fillPath(stroke, selected() ? MScore::selectColor[0] : MScore::layoutBreakColor);
painter->setPen(QPen(selected() ? MScore::selectColor[0] : MScore::layoutBreakColor,
lw, Qt::SolidLine, Qt::SquareCap, Qt::MiterJoin));
painter->setBrush(Qt::NoBrush);
painter->drawPath(path2);
}
//---------------------------------------------------------
// layout0
//---------------------------------------------------------
void LayoutBreak::layout0()
{
qreal _spatium = spatium();
path = QPainterPath();
path2 = QPainterPath();
qreal h = _spatium * 2.5;
qreal w = _spatium * 2.5;
QRectF rect(0.0, 0.0, w, h);
path.addRect(rect);
switch(layoutBreakType()) {
case LAYOUT_BREAK_LINE:
path2.moveTo(w * .8, h * .3);
path2.lineTo(w * .8, h * .6);
path2.lineTo(w * .3, h * .6);
path2.moveTo(w * .4, h * .5);
path2.lineTo(w * .25, h * .6);
path2.lineTo(w * .4, h * .7);
path2.lineTo(w * .4, h * .5);
break;
case LAYOUT_BREAK_PAGE:
path2.moveTo(w*.25, h*.2);
path2.lineTo(w*.60, h*.2);
path2.lineTo(w*.75, h*.35);
path2.lineTo(w*.75, h*.8);
path2.lineTo(w*.25, h*.8);
path2.lineTo(w*.25, h*.2);
path2.moveTo(w*.55, h*.21); // 0.01 to avoid overlap
path2.lineTo(w*.55, h*.40);
path2.lineTo(w*.74, h*.40);
break;
case LAYOUT_BREAK_SECTION:
path2.moveTo(w*.25, h*.2);
path2.lineTo(w*.75, h*.2);
path2.lineTo(w*.75, h*.8);
path2.lineTo(w*.25, h*.8);
path2.moveTo(w*.55, h*.21); // 0.01 to avoid overlap
path2.lineTo(w*.55, h*.79);
break;
default:
qDebug("unknown layout break symbol\n");
break;
}
QRectF bb(0, 0, w, h);
bb.adjust(-lw, -lw, lw, lw);
setbbox(bb);
}
//---------------------------------------------------------
// setLayoutBreakType
//---------------------------------------------------------
void LayoutBreak::setLayoutBreakType(LayoutBreakType val)
{
_layoutBreakType = val;
layout0();
}
//---------------------------------------------------------
// spatiumChanged
//---------------------------------------------------------
void LayoutBreak::spatiumChanged(qreal, qreal)
{
lw = spatium() * 0.3;
layout0();
}
//---------------------------------------------------------
// acceptDrop
//---------------------------------------------------------
bool LayoutBreak::acceptDrop(MuseScoreView*, const QPointF&, Element* e) const
{
return e->type() == LAYOUT_BREAK && static_cast<LayoutBreak*>(e)->layoutBreakType() != layoutBreakType();
}
//---------------------------------------------------------
// drop
//---------------------------------------------------------
Element* LayoutBreak::drop(const DropData& data)
{
Element* e = data.element;
score()->undoChangeElement(this, e);
return e;
}
//---------------------------------------------------------
// getProperty
//---------------------------------------------------------
QVariant LayoutBreak::getProperty(P_ID propertyId) const
{
switch(propertyId) {
case P_LAYOUT_BREAK:
return _layoutBreakType;
case P_PAUSE:
return _pause;
default:
return Element::getProperty(propertyId);
}
}
//---------------------------------------------------------
// setProperty
//---------------------------------------------------------
bool LayoutBreak::setProperty(P_ID propertyId, const QVariant& v)
{
switch(propertyId) {
case P_LAYOUT_BREAK:
setLayoutBreakType(LayoutBreakType(v.toInt()));
break;
case P_PAUSE:
setPause(v.toDouble());
break;
default:
if (!Element::setProperty(propertyId, v))
return false;
break;
}
score()->setLayoutAll(true);
setGenerated(false);
return true;
}
//---------------------------------------------------------
// propertyDefault
//---------------------------------------------------------
QVariant LayoutBreak::propertyDefault(P_ID id) const
{
switch(id) {
case P_LAYOUT_BREAK:
return QVariant(); // LAYOUT_BREAK_LINE;
case P_PAUSE:
return 0.0; // score()->styleD(ST_SectionPause);
default:
return Element::propertyDefault(id);
}
}
}