MuseScore/libmscore/measure.h

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

338 lines
12 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
//=============================================================================
#ifndef __MEASURE_H__
#define __MEASURE_H__
/**
\file
2016-12-28 16:23:10 +01:00
Definition of class Measure.
2012-05-26 14:26:10 +02:00
*/
#include "measurebase.h"
#include "fraction.h"
#include "segmentlist.h"
2013-05-13 18:49:17 +02:00
namespace Ms {
2016-11-19 11:51:21 +01:00
class XmlWriter;
2012-05-26 14:26:10 +02:00
class Beam;
class Tuplet;
class Staff;
class Chord;
2018-10-24 10:40:03 +02:00
class MeasureNumber;
2012-05-26 14:26:10 +02:00
class ChordRest;
class Score;
class MuseScoreView;
class System;
class Note;
class Spacer;
class TieMap;
class AccidentalState;
class Spanner;
class Part;
class RepeatMeasure;
2012-05-26 14:26:10 +02:00
2016-12-12 14:55:35 +01:00
class MStaff;
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
// MeasureNumberMode
//---------------------------------------------------------
enum class MeasureNumberMode : char {
AUTO, // show measure number depending on style
SHOW, // always show measure number
HIDE // dont show measure number
};
//---------------------------------------------------------
// MStaff
/// Per staff values of measure.
//---------------------------------------------------------
class MStaff
{
MeasureNumber* _noText { 0 }; ///< Measure number text object
StaffLines* _lines { 0 };
Spacer* _vspacerUp { 0 };
Spacer* _vspacerDown { 0 };
bool _hasVoices { false }; ///< indicates that MStaff contains more than one voice,
///< this changes some layout rules
bool _visible { true };
bool _stemless { false };
#ifndef NDEBUG
bool _corrupted { false };
#endif
public:
MStaff() {}
~MStaff();
MStaff(const MStaff&);
void setScore(Score*);
void setTrack(int);
MeasureNumber* noText() const { return _noText; }
void setNoText(MeasureNumber* t) { _noText = t; }
StaffLines* lines() const { return _lines; }
void setLines(StaffLines* l) { _lines = l; }
Spacer* vspacerUp() const { return _vspacerUp; }
void setVspacerUp(Spacer* s) { _vspacerUp = s; }
Spacer* vspacerDown() const { return _vspacerDown; }
void setVspacerDown(Spacer* s) { _vspacerDown = s; }
bool hasVoices() const { return _hasVoices; }
void setHasVoices(bool val) { _hasVoices = val; }
bool visible() const { return _visible; }
void setVisible(bool val) { _visible = val; }
bool stemless() const { return _stemless; }
void setStemless(bool val) { _stemless = val; }
#ifndef NDEBUG
bool corrupted() const { return _corrupted; }
void setCorrupted(bool val) { _corrupted = val; }
#endif
};
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
2012-07-11 21:29:42 +02:00
// @@ Measure
2012-07-25 11:49:34 +02:00
/// one measure in a system
Fixes the following Q_INVOKABLE methods returning a QObject* by turning them into a property: - Measure: -- firstSegment -- lastSegment - MeasureBase: -- nextMeasure -- nextMeasureMM (new) -- prevMeasure -- prevMeasureMM (new) - Score: -- firstMeasure -- firstMeasureMM (new) -- (for firstSegment(), see special cases below) -- lastMeasure -- lastMeasureMM (new) -- lastSegment - Segment: -- next (renamed from `next1`) -- nextInMeasure (renamed from `next`) -- prev (renamed from `prev1`) -- prevInMeasure (renamed from prev) Special cases: - Cursor: The prototype of the `Q_INVOKABLE Ms::Note* Cursor::addNote(int pitch)` was wrong: corrected in `Q_INVOKABLE void Cursor::addNote(int pitch)`. - QmlPlugin: `Q_INVOKABLE Score* QmlPlugin::readScore()` and `Q_INVOKABLE Score* QmlPlugin::newScore()` has been kept, as they are intended to be called from QML; code has been added to ensure the C++ ownership of the returned object. - Score: `Q_INVOKABLE Segment* Score::firstSegment(Segment::Type segType)` is kept (as it needs a parameters), but code is added to ensure C++ ownership of the returned Segment*. - Segment: `Ms::Element* Segment::element(int track)` has been made NOT Q_INVOKABLE; a variant `Q_INVOKABLE Ms::Element* elementAt(int track)` has been added specifically for QML with code to ensure the C++ ownership of the returned Element* (this was the cause for the crash of the Walk plug-in). - FiguredBass: `Q_INVOKABLE Ms::FiguredBassItem* FiguredBass::addItem()` has been removed; plugin interface for FiguredBass needs to be redesigned anyway. The few occurrences in the supplied plug-ins of the methods whose names did change have been updated.
2014-07-06 01:56:30 +02:00
//
// @P firstSegment Segment the first segment of the measure (read-only)
// @P lastSegment Segment the last segment of the measure (read-only)
2012-05-26 14:26:10 +02:00
//---------------------------------------------------------
class Measure final : public MeasureBase
{
2016-02-06 22:03:43 +01:00
std::vector<MStaff*> _mstaves;
2012-05-26 14:26:10 +02:00
SegmentList _segments;
2016-01-04 14:48:58 +01:00
Measure* _mmRest; // multi measure rest which replaces a measure range
qreal _userStretch;
2016-02-04 11:27:47 +01:00
Fraction _timesig;
2016-01-04 14:48:58 +01:00
int _mmRestCount; // > 0 if this is a multi measure rest
// 0 if this is the start of a mm rest (_mmRest != 0)
// < 0 if this measure is covered by a mm rest
int _playbackCount; // temp. value used in RepeatList
// counts how many times this measure was already played
2012-05-26 14:26:10 +02:00
2016-01-04 14:48:58 +01:00
int _repeatCount; ///< end repeat marker und repeat count
2012-05-26 14:26:10 +02:00
MeasureNumberMode _noMode;
bool _breakMultiMeasureRest;
void push_back(Segment* e);
2016-11-07 09:59:06 +01:00
void push_front(Segment* e);
void fillGap(const Fraction& pos, const Fraction& len, int track, const Fraction& stretch);
void computeMinWidth(Segment* s, qreal x, bool isSystemHeader);
2012-05-26 14:26:10 +02:00
void readVoice(XmlReader& e, int staffIdx, bool irregular);
2020-03-25 15:06:32 +01:00
2020-05-26 15:54:26 +02:00
public:
2020-03-25 15:06:32 +01:00
Measure(Score* = 0);
Measure(const Measure&);
~Measure();
2012-05-26 14:26:10 +02:00
2020-03-25 15:06:32 +01:00
Measure* clone() const override { return new Measure(*this); }
ElementType type() const override { return ElementType::MEASURE; }
void setScore(Score* s) override;
Measure* cloneMeasure(Score*, const Fraction& tick, TieMap*);
2012-05-26 14:26:10 +02:00
// Score Tree functions
ScoreElement* treeParent() const override;
ScoreElement* treeChild(int idx) const override;
int treeChildCount() const override;
2020-03-25 15:06:32 +01:00
void read(XmlReader&, int idx);
void read(XmlReader& d) { read(d, 0); }
void readAddConnector(ConnectorInfoReader* info, bool pasteMode) override;
void write(XmlWriter& xml) const override { Element::write(xml); }
void write(XmlWriter&, int, bool writeSystemElements, bool forceTimeSig) const;
2016-11-19 11:51:21 +01:00
void writeBox(XmlWriter&) const;
2013-01-11 18:10:18 +01:00
void readBox(XmlReader&);
2020-03-25 15:06:32 +01:00
bool isEditable() const override { return false; }
void checkMeasure(int idx);
2012-05-26 14:26:10 +02:00
2016-12-12 12:02:18 +01:00
void add(Element*) override;
void remove(Element*) override;
void change(Element* o, Element* n) override;
2016-12-12 14:55:35 +01:00
void spatiumChanged(qreal oldValue, qreal newValue) override;
System* system() const { return (System*)parent(); }
bool hasVoices(int staffIdx, Fraction stick, Fraction len) const;
bool hasVoices(int staffIdx) const;
void setHasVoices(int staffIdx, bool v);
2016-12-12 14:55:35 +01:00
StaffLines* staffLines(int staffIdx);
Spacer* vspacerDown(int staffIdx) const;
Spacer* vspacerUp(int staffIdx) const;
void setStaffVisible(int staffIdx, bool visible);
void setStaffStemless(int staffIdx, bool stemless);
bool corrupted(int staffIdx) const;
void setCorrupted(int staffIdx, bool val);
void setNoText(int staffIdx, MeasureNumber*);
MeasureNumber* noText(int staffIdx) const;
2012-05-26 14:26:10 +02:00
void createStaves(int);
2020-05-26 15:54:26 +02:00
2012-05-26 14:26:10 +02:00
MeasureNumberMode measureNumberMode() const { return _noMode; }
void setMeasureNumberMode(MeasureNumberMode v) { _noMode = v; }
Fraction timesig() const { return _timesig; }
void setTimesig(const Fraction& f) { _timesig = f; }
2013-05-13 18:49:17 +02:00
2012-05-26 14:26:10 +02:00
Fraction stretchedLen(Staff*) const;
2016-08-06 11:36:51 +02:00
bool isIrregular() const { return _timesig != _len; }
2020-05-26 15:54:26 +02:00
int size() const { return _segments.size(); }
Ms::Segment* first() const { return _segments.first(); }
Segment* first(SegmentType t) const { return _segments.first(t); }
Segment* firstEnabled() const { return _segments.first(ElementFlag::ENABLED); }
2020-05-26 15:54:26 +02:00
Ms::Segment* last() const { return _segments.last(); }
SegmentList& segments() { return _segments; }
const SegmentList& segments() const { return _segments; }
2020-05-26 15:54:26 +02:00
2014-05-17 18:40:05 +02:00
qreal userStretch() const;
void setUserStretch(qreal v) { _userStretch = v; }
2020-05-26 15:54:26 +02:00
2016-01-04 14:48:58 +01:00
void stretchMeasure(qreal stretch);
Fraction computeTicks();
2012-05-26 14:26:10 +02:00
void layout2();
2020-05-26 15:54:26 +02:00
2020-03-10 07:50:48 +01:00
bool showsMeasureNumber();
bool showsMeasureNumberInAutoMode();
void layoutMeasureNumber();
2020-05-26 15:54:26 +02:00
Chord* findChord(Fraction tick, int track);
ChordRest* findChordRest(Fraction tick, int track);
Fraction snap(const Fraction& tick, const QPointF p) const;
Fraction snapNote(const Fraction& tick, const QPointF p, int staff) const;
2020-05-26 15:54:26 +02:00
Segment* searchSegment(qreal x, SegmentType st, int strack, int etrack, const Segment* preferredSegment = nullptr,
qreal spacingFactor = 0.5) const;
2020-05-26 15:54:26 +02:00
2012-05-26 14:26:10 +02:00
void insertStaff(Staff*, int staff);
void insertMStaff(MStaff* staff, int idx);
void removeMStaff(MStaff* staff, int idx);
2020-05-26 15:54:26 +02:00
2020-03-25 15:06:32 +01:00
void moveTicks(const Fraction& diff) override;
2020-05-26 15:54:26 +02:00
2012-05-26 14:26:10 +02:00
void cmdRemoveStaves(int s, int e);
void cmdAddStaves(int s, int e, bool createRest);
void removeStaves(int s, int e);
void insertStaves(int s, int e);
2020-05-26 15:54:26 +02:00
qreal tick2pos(Fraction) const;
Segment* tick2segment(const Fraction& tick, SegmentType st = SegmentType::ChordRest);
2020-05-26 15:54:26 +02:00
2012-05-26 14:26:10 +02:00
void sortStaves(QList<int>& dst);
2020-05-26 15:54:26 +02:00
2020-03-25 15:06:32 +01:00
bool acceptDrop(EditData&) const override;
Element* drop(EditData&) override;
2020-05-26 15:54:26 +02:00
2012-05-26 14:26:10 +02:00
int repeatCount() const { return _repeatCount; }
void setRepeatCount(int val) { _repeatCount = val; }
2020-05-26 15:54:26 +02:00
Segment* findSegmentR(SegmentType st, const Fraction&) const;
Segment* undoGetSegmentR(SegmentType st, const Fraction& f);
Segment* getSegmentR(SegmentType st, const Fraction& f);
Segment* findFirstR(SegmentType st, const Fraction& rtick) const;
2020-05-26 15:54:26 +02:00
// segment routines with absolute tick values
Segment* findSegment(SegmentType st, const Fraction& f) const { return findSegmentR(st, f - tick()); }
Segment* undoGetSegment(SegmentType st, const Fraction& f) { return undoGetSegmentR(st, f - tick()); }
Segment* getSegment(SegmentType st, const Fraction& f) { return getSegmentR(st, f - tick()); }
2020-05-26 15:54:26 +02:00
void connectTremolo();
2020-05-26 15:54:26 +02:00
2016-01-04 14:48:58 +01:00
qreal createEndBarLines(bool);
2016-10-18 15:41:00 +02:00
void barLinesSetSpan(Segment*);
void setEndBarLineType(BarLineType val, int track, bool visible = true, QColor color = QColor());
2020-05-26 15:54:26 +02:00
RepeatMeasure* cmdInsertRepeatMeasure(int staffIdx);
2020-05-26 15:54:26 +02:00
2020-03-25 15:06:32 +01:00
void scanElements(void* data, void (* func)(void*, Element*), bool all=true) override;
2012-05-26 14:26:10 +02:00
void createVoice(int track);
void adjustToLen(Fraction, bool appendRestsIfNecessary = true);
2020-05-26 15:54:26 +02:00
2012-08-08 20:46:29 +02:00
AccidentalVal findAccidental(Note*) const;
AccidentalVal findAccidental(Segment* s, int staffIdx, int line, bool& error) const;
2014-08-29 10:35:17 +02:00
void exchangeVoice(int voice1, int voice2, int staffIdx);
2012-05-26 14:26:10 +02:00
void checkMultiVoices(int staffIdx);
bool hasVoice(int track) const;
bool isEmpty(int staffIdx) const;
2015-08-04 19:53:54 +02:00
bool isFullMeasureRest() const;
bool isRepeatMeasure(const Staff* staff) const;
2012-05-26 14:26:10 +02:00
bool visible(int staffIdx) const;
bool stemless(int staffIdx) const;
bool isFinalMeasureOfSection() const;
bool isAnacrusis() const;
bool isFirstInSystem() const;
2020-05-26 15:54:26 +02:00
2016-01-04 14:48:58 +01:00
bool breakMultiMeasureRest() const { return _breakMultiMeasureRest; }
2012-05-26 14:26:10 +02:00
void setBreakMultiMeasureRest(bool val) { _breakMultiMeasureRest = val; }
2020-05-26 15:54:26 +02:00
2016-02-06 22:03:43 +01:00
bool empty() const;
bool isOnlyRests(int track) const;
bool isOnlyDeletedRests(int track) const;
2020-05-26 15:54:26 +02:00
2012-05-26 14:26:10 +02:00
int playbackCount() const { return _playbackCount; }
void setPlaybackCount(int val) { _playbackCount = val; }
QRectF staffabbox(int staffIdx) const;
2020-05-26 15:54:26 +02:00
2020-03-25 15:06:32 +01:00
QVariant getProperty(Pid propertyId) const override;
bool setProperty(Pid propertyId, const QVariant&) override;
QVariant propertyDefault(Pid) const override;
2020-05-26 15:54:26 +02:00
bool hasMMRest() const { return _mmRest != 0; }
2013-09-27 18:43:25 +02:00
bool isMMRest() const { return _mmRestCount > 0; }
Measure* mmRest() const { return _mmRest; }
const Measure* mmRest1() const;
void setMMRest(Measure* m) { _mmRest = m; }
int mmRestCount() const { return _mmRestCount; } // number of measures _mmRest spans
void setMMRestCount(int n) { _mmRestCount = n; }
2013-10-30 14:21:08 +01:00
Measure* mmRestFirst() const;
Measure* mmRestLast() const;
2020-05-26 15:54:26 +02:00
Element* nextElementStaff(int staff);
Element* prevElementStaff(int staff);
2020-03-25 15:06:32 +01:00
QString accessibleInfo() const override;
2020-05-26 15:54:26 +02:00
2016-10-18 15:41:00 +02:00
void addSystemHeader(bool firstSystem);
void addSystemTrailer(Measure* nm);
void removeSystemHeader();
void removeSystemTrailer();
2020-05-26 15:54:26 +02:00
const BarLine* endBarLine() const;
2016-04-18 09:46:30 +02:00
BarLineType endBarLineType() const;
bool endBarLineVisible() const;
2020-03-25 15:06:32 +01:00
void triggerLayout() const override;
2016-10-18 15:41:00 +02:00
qreal basicStretch() const;
qreal basicWidth() const;
int layoutWeight(int maxMMRestLength = 0) const;
2020-03-25 15:06:32 +01:00
void computeMinWidth();
void checkHeader();
void checkTrailer();
2016-10-31 10:55:11 +01:00
void setStretchedWidth(qreal);
2016-12-12 14:55:35 +01:00
void layoutStaffLines();
2020-05-26 15:54:26 +02:00
};
2013-05-13 18:49:17 +02:00
} // namespace Ms
2012-05-26 14:26:10 +02:00
#endif