601 lines
19 KiB
C++
601 lines
19 KiB
C++
//=============================================================================
|
|
// MuseScore
|
|
// Music Composition & Notation
|
|
//
|
|
// Copyright (C) 2010-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 "fret.h"
|
|
#include "measure.h"
|
|
#include "system.h"
|
|
#include "score.h"
|
|
#include "stringdata.h"
|
|
#include "chord.h"
|
|
#include "note.h"
|
|
#include "segment.h"
|
|
#include "mscore.h"
|
|
#include "harmony.h"
|
|
|
|
namespace Ms {
|
|
|
|
static const int DEFAULT_STRINGS = 6;
|
|
static const int DEFAULT_FRETS = 5;
|
|
|
|
// parent() is Segment or Box
|
|
//
|
|
|
|
//---------------------------------------------------------
|
|
// FretDiagram
|
|
//---------------------------------------------------------
|
|
|
|
FretDiagram::FretDiagram(Score* score)
|
|
: Element(score)
|
|
{
|
|
setFlags(ElementFlag::MOVABLE | ElementFlag::ON_STAFF | ElementFlag::SELECTABLE);
|
|
_strings = DEFAULT_STRINGS;
|
|
_frets = DEFAULT_FRETS;
|
|
_maxFrets = 24;
|
|
maxStrings = 0;
|
|
_dots = 0;
|
|
_marker = 0;
|
|
_fingering = 0;
|
|
_fretOffset = 0;
|
|
font.setFamily("FreeSans");
|
|
int size = lrint(4.0 * MScore::DPI * mag()/ PPI);
|
|
font.setPixelSize(size);
|
|
_harmony = 0;
|
|
}
|
|
|
|
FretDiagram::FretDiagram(const FretDiagram& f)
|
|
: Element(f)
|
|
{
|
|
_strings = f._strings;
|
|
_frets = f._frets;
|
|
_fretOffset = f._fretOffset;
|
|
_maxFrets = f._maxFrets;
|
|
maxStrings = f.maxStrings;
|
|
_dots = 0;
|
|
_marker = 0;
|
|
_fingering = 0;
|
|
font = f.font;
|
|
|
|
if (f._dots) {
|
|
_dots = new char[_strings];
|
|
memcpy(_dots, f._dots, _strings);
|
|
}
|
|
if (f._marker) {
|
|
_marker = new char[_strings];
|
|
memcpy(_marker, f._marker, _strings);
|
|
}
|
|
if (f._fingering) {
|
|
_fingering = new char[_strings];
|
|
memcpy(_fingering, f._fingering, _strings);
|
|
}
|
|
if (f._harmony)
|
|
_harmony = new Harmony(*f._harmony);
|
|
else
|
|
_harmony = 0;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// FretDiagram
|
|
//---------------------------------------------------------
|
|
|
|
FretDiagram::~FretDiagram()
|
|
{
|
|
delete[] _dots;
|
|
delete[] _marker;
|
|
delete[] _fingering;
|
|
}
|
|
|
|
#if 1
|
|
//---------------------------------------------------------
|
|
// pagePos
|
|
//---------------------------------------------------------
|
|
|
|
QPointF FretDiagram::pagePos() const
|
|
{
|
|
if (parent() == 0)
|
|
return pos();
|
|
if (parent()->type() == ElementType::SEGMENT) {
|
|
Measure* m = static_cast<Segment*>(parent())->measure();
|
|
System* system = m->system();
|
|
qreal yp = y();
|
|
if (system)
|
|
yp += system->staffYpage(staffIdx());
|
|
return QPointF(pageX(), yp);
|
|
}
|
|
else
|
|
return Element::pagePos();
|
|
}
|
|
#endif
|
|
|
|
//---------------------------------------------------------
|
|
// dragAnchor
|
|
//---------------------------------------------------------
|
|
|
|
QLineF FretDiagram::dragAnchor() const
|
|
{
|
|
if (parent()->type() == ElementType::SEGMENT) {
|
|
Segment* s = static_cast<Segment*>(parent());
|
|
Measure* m = s->measure();
|
|
System* system = m->system();
|
|
qreal yp = system->staff(staffIdx())->y() + system->y();
|
|
qreal xp = m->tick2pos(s->tick()) + m->pagePos().x();
|
|
QPointF p1(xp, yp);
|
|
|
|
qreal x = 0.0;
|
|
qreal y = 0.0;
|
|
#if 0 // TODOxx
|
|
qreal tw = width();
|
|
qreal th = height();
|
|
if (_align & ALIGN_BOTTOM)
|
|
y = th;
|
|
else if (_align & ALIGN_VCENTER)
|
|
y = (th * .5);
|
|
else if (_align & ALIGN_BASELINE)
|
|
y = baseLine();
|
|
if (_align & ALIGN_RIGHT)
|
|
x = tw;
|
|
else if (_align & ALIGN_HCENTER)
|
|
x = (tw * .5);
|
|
#endif
|
|
return QLineF(p1, abbox().topLeft() + QPointF(x, y));
|
|
}
|
|
return QLineF(parent()->pagePos(), abbox().topLeft());
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// setOffset
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::setOffset(int offset)
|
|
{
|
|
_fretOffset = offset;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// setStrings
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::setStrings(int n)
|
|
{
|
|
if (n <= maxStrings) {
|
|
_strings = n;
|
|
return;
|
|
}
|
|
maxStrings = n;
|
|
if (_dots) {
|
|
char* ndots = new char[n];
|
|
memcpy(ndots, _dots, n);
|
|
for (int i = _strings; i < n; ++i)
|
|
ndots[i] = 0;
|
|
delete[] _dots;
|
|
_dots = ndots;
|
|
}
|
|
if (_marker) {
|
|
char* nmarker = new char[n];
|
|
memcpy(nmarker, _marker, n);
|
|
for (int i = _strings; i < n; ++i)
|
|
nmarker[i] = 'O';
|
|
delete[] _marker;
|
|
_marker = nmarker;
|
|
}
|
|
if (_fingering) {
|
|
char* nfingering = new char[n];
|
|
memcpy(nfingering, _fingering, n);
|
|
for (int i = _strings; i < n; ++i)
|
|
nfingering[i] = 0;
|
|
delete[] _fingering;
|
|
_fingering = nfingering;
|
|
}
|
|
_strings = n;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// init
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::init(StringData* stringData, Chord* chord)
|
|
{
|
|
if (stringData == 0)
|
|
setStrings(6);
|
|
else
|
|
setStrings(stringData->strings());
|
|
if (stringData) {
|
|
for (int string = 0; string < _strings; ++string)
|
|
_marker[string] = 'X';
|
|
foreach(const Note* note, chord->notes()) {
|
|
int string;
|
|
int fret;
|
|
if (stringData->convertPitch(note->ppitch(), &string, &fret))
|
|
setDot(string, fret);
|
|
}
|
|
_maxFrets = stringData->frets();
|
|
}
|
|
else
|
|
_maxFrets = 6;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// draw
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::draw(QPainter* painter) const
|
|
{
|
|
qreal _spatium = spatium();
|
|
QPen pen(curColor());
|
|
pen.setWidthF(lw2);
|
|
pen.setCapStyle(Qt::FlatCap);
|
|
painter->setPen(pen);
|
|
painter->setBrush(QBrush(QColor(painter->pen().color())));
|
|
qreal x2 = (_strings-1) * stringDist;
|
|
painter->drawLine(QLineF(-lw1*.5, 0.0, x2+lw1*.5, 0.0));
|
|
|
|
pen.setWidthF(lw1);
|
|
painter->setPen(pen);
|
|
qreal y2 = (_frets+1) * fretDist - fretDist*.5;
|
|
for (int i = 0; i < _strings; ++i) {
|
|
qreal x = stringDist * i;
|
|
painter->drawLine(QLineF(x, _fretOffset ? -_spatium*.2 : 0.0, x, y2));
|
|
}
|
|
for (int i = 1; i <= _frets; ++i) {
|
|
qreal y = fretDist * i;
|
|
painter->drawLine(QLineF(0.0, y, x2, y));
|
|
}
|
|
painter->setFont(font);
|
|
QFontMetricsF fm(font);
|
|
for (int i = 0; i < _strings; ++i) {
|
|
if (_dots && _dots[i]) {
|
|
qreal dotd = stringDist * .6;
|
|
int fret = _dots[i] - 1;
|
|
qreal x = stringDist * i - dotd * .5;
|
|
qreal y = fretDist * fret + fretDist * .5 - dotd * .5;
|
|
painter->drawEllipse(QRectF(x, y, dotd, dotd));
|
|
}
|
|
if (_marker && _marker[i]) {
|
|
qreal x = stringDist * i;
|
|
qreal y = -fretDist * .3 - fm.ascent();
|
|
painter->drawText(QRectF(x, y, .0,.0),
|
|
Qt::AlignHCenter|Qt::TextDontClip, QChar(_marker[i]));
|
|
}
|
|
}
|
|
if (_fretOffset > 0) {
|
|
qreal fretNumMag = score()->styleD(StyleIdx::fretNumMag);
|
|
QFont scaledFont(font);
|
|
scaledFont.setPixelSize(font.pixelSize() * fretNumMag);
|
|
painter->setFont(scaledFont);
|
|
if ( score()->styleI(StyleIdx::fretNumPos) == 0 )
|
|
painter->drawText(QRectF(-stringDist *.4, .0, .0, fretDist),
|
|
Qt::AlignVCenter|Qt::AlignRight|Qt::TextDontClip,
|
|
QString("%1").arg(_fretOffset+1));
|
|
else
|
|
painter->drawText(QRectF(x2 + (stringDist * 0.4), .0, .0, fretDist),
|
|
Qt::AlignVCenter|Qt::AlignLeft|Qt::TextDontClip,
|
|
QString("%1").arg(_fretOffset+1));
|
|
painter->setFont(font);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// layout
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::layout()
|
|
{
|
|
qreal _spatium = spatium();
|
|
lw1 = _spatium * 0.08;
|
|
lw2 = _fretOffset ? lw1 : _spatium * 0.2;
|
|
stringDist = _spatium * .7;
|
|
fretDist = _spatium * .8;
|
|
|
|
qreal w = stringDist * (_strings-1);
|
|
qreal h = _frets * fretDist + fretDist * .5;
|
|
qreal y = 0.0;
|
|
qreal dotd = stringDist * .6;
|
|
qreal x = -((dotd+lw1) * .5);
|
|
w += dotd + lw1;
|
|
if (_marker) {
|
|
QFontMetricsF fm(font);
|
|
y = -(fretDist * .1 + fm.height());
|
|
h -= y;
|
|
}
|
|
bbox().setRect(x, y, w, h);
|
|
|
|
setPos(-_spatium, -h - score()->styleP(StyleIdx::fretY) + _spatium );
|
|
adjustReadPos();
|
|
|
|
if (_harmony)
|
|
_harmony->layout();
|
|
|
|
if (parent() == 0 || parent()->type() != ElementType::SEGMENT)
|
|
return;
|
|
Measure* m = static_cast<Segment*>(parent())->measure();
|
|
int idx = staffIdx();
|
|
MStaff* mstaff = m->mstaff(idx);
|
|
qreal dist = -(bbox().top());
|
|
mstaff->distanceUp = qMax(mstaff->distanceUp, dist + _spatium * 2);
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// write
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::write(Xml& xml) const
|
|
{
|
|
xml.stag("FretDiagram");
|
|
Element::writeProperties(xml);
|
|
|
|
if (_strings != DEFAULT_STRINGS)
|
|
xml.tag("strings", _strings);
|
|
if (_frets != DEFAULT_FRETS)
|
|
xml.tag("frets", _frets);
|
|
if (_fretOffset)
|
|
xml.tag("fretOffset", _fretOffset);
|
|
for (int i = 0; i < _strings; ++i) {
|
|
if ((_dots && _dots[i]) || (_marker && _marker[i]) || (_fingering && _fingering[i])) {
|
|
xml.stag(QString("string no=\"%1\"").arg(i));
|
|
if (_dots && _dots[i])
|
|
xml.tag("dot", _dots[i]);
|
|
if (_marker && _marker[i])
|
|
xml.tag("marker", _marker[i]);
|
|
if (_fingering && _fingering[i])
|
|
xml.tag("fingering", _fingering[i]);
|
|
xml.etag();
|
|
}
|
|
}
|
|
if (_harmony)
|
|
_harmony->write(xml);
|
|
xml.etag();
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// read
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::read(XmlReader& e)
|
|
{
|
|
while (e.readNextStartElement()) {
|
|
const QStringRef& tag(e.name());
|
|
if (tag == "strings")
|
|
_strings = e.readInt();
|
|
else if (tag == "frets")
|
|
_frets = e.readInt();
|
|
else if (tag == "fretOffset")
|
|
_fretOffset = e.readInt();
|
|
else if (tag == "string") {
|
|
int no = e.intAttribute("no");
|
|
while (e.readNextStartElement()) {
|
|
const QStringRef& tag(e.name());
|
|
if (tag == "dot")
|
|
setDot(no, e.readInt());
|
|
else if (tag == "marker")
|
|
setMarker(no, e.readInt());
|
|
else if (tag == "fingering")
|
|
setFingering(no, e.readInt());
|
|
else
|
|
e.unknown();
|
|
}
|
|
}
|
|
else if (tag == "Harmony") {
|
|
Harmony* h = new Harmony(score());
|
|
h->read(e);
|
|
add(h);
|
|
}
|
|
else if (!Element::readProperties(e))
|
|
e.unknown();
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// setDot
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::setDot(int string, int fret)
|
|
{
|
|
if (_dots == 0) {
|
|
_dots = new char[_strings];
|
|
memset(_dots, 0, _strings);
|
|
}
|
|
if (0 <= string && string < _strings) {
|
|
_dots[string] = fret;
|
|
setMarker(string, 0);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// setMarker
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::setMarker(int string, int marker)
|
|
{
|
|
if (_marker == 0) {
|
|
_marker = new char[_strings];
|
|
memset(_marker, 0, _strings);
|
|
}
|
|
if (0 <= string && string < _strings)
|
|
_marker[string] = marker;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// setFingering
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::setFingering(int string, int finger)
|
|
{
|
|
if (_fingering == 0) {
|
|
_fingering = new char[_strings];
|
|
memset(_fingering, 0, _strings);
|
|
}
|
|
_fingering[string] = finger;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// add
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::add(Element* e)
|
|
{
|
|
e->setParent(this);
|
|
if (e->type() == ElementType::HARMONY) {
|
|
_harmony = static_cast<Harmony*>(e);
|
|
_harmony->setTrack(track());
|
|
}
|
|
else
|
|
qWarning("FretDiagram: cannot add <%s>\n", e->name());
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// remove
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::remove(Element* e)
|
|
{
|
|
if (e == _harmony)
|
|
_harmony = 0;
|
|
else
|
|
qWarning("FretDiagram: cannot remove <%s>\n", e->name());
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// acceptDrop
|
|
//---------------------------------------------------------
|
|
|
|
bool FretDiagram::acceptDrop(MuseScoreView*, const QPointF&, Element* e) const
|
|
{
|
|
return e->type() == ElementType::HARMONY;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// drop
|
|
//---------------------------------------------------------
|
|
|
|
Element* FretDiagram::drop(const DropData& data)
|
|
{
|
|
Element* e = data.element;
|
|
if (e->type() == ElementType::HARMONY) {
|
|
// TODO: make undoable
|
|
Harmony* h = static_cast<Harmony*>(e);
|
|
h->setParent(this);
|
|
score()->undoAddElement(h);
|
|
}
|
|
else {
|
|
qWarning("FretDiagram: cannot drop <%s>\n", e->name());
|
|
delete e;
|
|
e = 0;
|
|
}
|
|
return e;
|
|
}
|
|
|
|
//---------------------------------------------------------
|
|
// scanElements
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::scanElements(void* data, void (*func)(void*, Element*), bool all)
|
|
{
|
|
Q_UNUSED(all);
|
|
func(data, this);
|
|
if (_harmony)
|
|
func(data, _harmony);
|
|
}
|
|
|
|
#if 0
|
|
//---------------------------------------------------------
|
|
// Read MusicXML
|
|
//
|
|
// Set the FretDiagram state based on the MusicXML <figure> node de.
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::readMusicXML(XmlReader& e)
|
|
{
|
|
qDebug("FretDiagram::readMusicXML");
|
|
|
|
while (e.readNextStartElement()) {
|
|
const QStringRef& tag(e.name());
|
|
if (tag == "frame-frets") {
|
|
int val = e.readInt();
|
|
if (val > 0)
|
|
setFrets(val);
|
|
else
|
|
qDebug("FretDiagram::readMusicXML: illegal frame-fret %d", val);
|
|
}
|
|
else if (tag == "frame-note") {
|
|
int fret = -1;
|
|
int string = -1;
|
|
while (e.readNextStartElement()) {
|
|
const QStringRef& tag(e.name());
|
|
int val = e.readInt();
|
|
if (tag == "fret")
|
|
fret = val;
|
|
else if (tag == "string")
|
|
string = val;
|
|
else
|
|
e.unknown();
|
|
}
|
|
qDebug("FretDiagram::readMusicXML string %d fret %d", string, fret);
|
|
if (string > 0) {
|
|
if (fret == 0)
|
|
setMarker(strings() - string, 79 /* ??? */);
|
|
else if (fret > 0)
|
|
setDot(strings() - string, fret);
|
|
}
|
|
}
|
|
else if (tag == "frame-strings") {
|
|
int val = e.readInt();
|
|
if (val > 0) {
|
|
setStrings(val);
|
|
for (int i = 0; i < val; ++i)
|
|
setMarker(i, 88 /* ??? */);
|
|
}
|
|
else
|
|
qDebug("FretDiagram::readMusicXML: illegal frame-strings %d", val);
|
|
}
|
|
else
|
|
e.unknown();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
//---------------------------------------------------------
|
|
// Write MusicXML
|
|
//---------------------------------------------------------
|
|
|
|
void FretDiagram::writeMusicXML(Xml& xml) const
|
|
{
|
|
qDebug("FretDiagram::writeMusicXML() this %p harmony %p", this, _harmony);
|
|
xml.stag("frame");
|
|
xml.tag("frame-strings", strings());
|
|
xml.tag("frame-frets", frets());
|
|
QString strDots = "'";
|
|
QString strMarker = "'";
|
|
QString strFingering = "'";
|
|
for (int i = 0; i < strings(); ++i) {
|
|
// TODO print frame note
|
|
if (_dots) strDots += QString("%1'").arg(static_cast<int>(_dots[i]));
|
|
if (_marker) strMarker += QString("%1'").arg(static_cast<int>(_marker[i]));
|
|
if (_fingering) strFingering += QString("%1'").arg(static_cast<int>(_fingering[i]));
|
|
if (_marker[i] != 88) {
|
|
xml.stag("frame-note");
|
|
xml.tag("string", strings() - i);
|
|
xml.tag("fret", _dots[i]);
|
|
xml.etag();
|
|
}
|
|
}
|
|
qDebug("FretDiagram::writeMusicXML() this %p dots %s marker %s fingering %s",
|
|
this, qPrintable(strDots), qPrintable(strMarker), qPrintable(strFingering));
|
|
/*
|
|
xml.tag("root-step", tpc2stepName(rootTpc));
|
|
int alter = tpc2alter(rootTpc);
|
|
if (alter)
|
|
xml.tag("root-alter", alter);
|
|
*/
|
|
xml.etag();
|
|
}
|
|
|
|
}
|
|
|