MuseScore/libmscore/cursor.cpp

287 lines
8.1 KiB
C++
Raw Normal View History

//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2012 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
//=============================================================================
2012-06-05 17:15:46 +02:00
#include "libmscore/score.h"
#include "libmscore/chordrest.h"
#include "libmscore/chord.h"
#include "libmscore/rest.h"
#include "libmscore/note.h"
#include "libmscore/stafftext.h"
#include "libmscore/measure.h"
#include "libmscore/repeatlist.h"
#include "libmscore/page.h"
#include "libmscore/system.h"
#include "libmscore/segment.h"
#include "libmscore/timesig.h"
#include "cursor.h"
2013-05-13 18:49:17 +02:00
namespace Ms {
//---------------------------------------------------------
// Cursor
//---------------------------------------------------------
Cursor::Cursor(Score* s)
: QObject(0)
{
2013-07-10 10:33:05 +02:00
_track = 0;
_segment = 0;
setScore(s);
}
void Cursor::setScore(Score* s)
{
_score = s;
if (_score) {
_score->inputState().setTrack(_track);
_score->inputState().setSegment(_segment);
}
}
//---------------------------------------------------------
// rewind
//---------------------------------------------------------
void Cursor::rewind(int type)
{
if (type == 0) {
2012-06-08 19:06:52 +02:00
_segment = 0;
Measure* m = _score->firstMeasure();
if (m) {
_segment = m->first(Segment::SegChordRest);
firstChordRestInTrack();
}
}
else if (type == 1) {
_segment = _score->selection().startSegment();
_track = _score->selection().staffStart() * VOICES;
firstChordRestInTrack();
}
else if (type == 2) {
_segment = _score->selection().endSegment();
_track = (_score->selection().staffEnd() * VOICES) - 1; // be sure _track exists
}
2012-06-08 19:06:52 +02:00
_score->inputState().setTrack(_track);
_score->inputState().setSegment(_segment);
}
//---------------------------------------------------------
// next
// go to next segment
// return false if end of score is reached
//---------------------------------------------------------
bool Cursor::next()
{
if (!_segment)
return false;
2013-06-12 14:23:57 +02:00
_segment = _segment->next1(Segment::SegChordRest);
firstChordRestInTrack();
2012-06-08 19:06:52 +02:00
_score->inputState().setTrack(_track);
_score->inputState().setSegment(_segment);
return _segment != 0;
}
//---------------------------------------------------------
// nextMeasure
// go to first segment of next measure
// return false if end of score is reached
//---------------------------------------------------------
bool Cursor::nextMeasure()
{
if (_segment == 0)
return false;
Measure* m = _segment->measure()->nextMeasure();
if (m == 0) {
_segment = 0;
return false;
}
2013-06-12 14:23:57 +02:00
_segment = m->first(Segment::SegChordRest);
// while (seg && seg->element(_track) == 0)
2013-06-12 14:23:57 +02:00
// seg = seg->next1(SegChordRest);
firstChordRestInTrack();
return _segment != 0;
}
//---------------------------------------------------------
// add
//---------------------------------------------------------
void Cursor::add(Element* s)
{
if (!_segment)
return;
2014-04-25 14:26:32 +02:00
s->setTrack(_track);
2012-06-07 09:24:05 +02:00
s->setParent(_segment);
2014-04-25 14:26:32 +02:00
if (s->isChordRest())
2012-06-07 09:24:05 +02:00
s->score()->undoAddCR(static_cast<ChordRest*>(s), _segment->measure(), _segment->tick());
else if (s->type() == Element::KEYSIG) {
Segment* ns = _segment->measure()->undoGetSegment(Segment::SegKeySig, _segment->tick());
s->setParent(ns);
_score->undoAddElement(s);
}
else if (s->type() == Element::TIMESIG) {
Measure* m = _segment->measure();
int tick = m->tick();
_score->cmdAddTimeSig(m, _track, static_cast<TimeSig*>(s), false);
m = _score->tick2measure(tick);
_segment = m->first(Segment::SegChordRest);
firstChordRestInTrack();
}
2013-10-05 23:13:33 +02:00
else if (s->type() == Element::LAYOUT_BREAK) {
Measure* m = _segment->measure();
s->setParent(m);
_score->undoAddElement(s);
}
else {
_score->undoAddElement(s);
2012-06-07 09:24:05 +02:00
}
_score->setLayoutAll(true);
}
//---------------------------------------------------------
2012-06-08 19:06:52 +02:00
// addNote
//---------------------------------------------------------
Note* Cursor::addNote(int pitch)
{
return _score->addPitch(pitch, false);
}
//---------------------------------------------------------
2012-06-08 19:06:52 +02:00
// setDuration
//---------------------------------------------------------
2012-06-08 19:06:52 +02:00
void Cursor::setDuration(int z, int n)
{
2012-06-08 19:06:52 +02:00
TDuration d(Fraction(z, n));
if (!d.isValid())
d = TDuration(TDuration::DurationType::V_QUARTER);
2012-06-08 19:06:52 +02:00
_score->inputState().setDuration(d);
}
//---------------------------------------------------------
2012-06-08 19:06:52 +02:00
// tick
//---------------------------------------------------------
2012-06-08 19:06:52 +02:00
int Cursor::tick()
{
2012-06-08 19:06:52 +02:00
return (_segment) ? _segment->tick() : 0;
}
2012-09-13 16:10:13 +02:00
//---------------------------------------------------------
// time
//---------------------------------------------------------
double Cursor::time()
{
return _score->utick2utime(tick()) * 1000;
}
//---------------------------------------------------------
// element
//---------------------------------------------------------
Element* Cursor::element() const
{
return _segment ? _segment->element(_track) : 0;
}
//---------------------------------------------------------
// measure
//---------------------------------------------------------
Measure* Cursor::measure() const
{
return _segment ? _segment->measure() : 0;
}
//---------------------------------------------------------
// setTrack
//---------------------------------------------------------
void Cursor::setTrack(int v)
{
_track = v;
int tracks = _score->nstaves() * VOICES;
if (_track < 0)
_track = 0;
else if (_track >= tracks)
_track = tracks - 1;
2012-06-08 19:06:52 +02:00
_score->inputState().setTrack(_track);
}
//---------------------------------------------------------
// setStaffIdx
//---------------------------------------------------------
void Cursor::setStaffIdx(int v)
{
_track = v * VOICES + _track % VOICES;
int tracks = _score->nstaves() * VOICES;
if (_track < 0)
_track = 0;
else if (_track >= tracks)
_track = tracks - 1;
2012-06-08 19:06:52 +02:00
_score->inputState().setTrack(_track);
}
//---------------------------------------------------------
// setVoice
//---------------------------------------------------------
void Cursor::setVoice(int v)
{
_track = (_track / VOICES) * VOICES + v;
int tracks = _score->nstaves() * VOICES;
if (_track < 0)
_track = 0;
else if (_track >= tracks)
_track = tracks - 1;
2012-06-08 19:06:52 +02:00
_score->inputState().setTrack(_track);
}
//---------------------------------------------------------
// staffIdx
//---------------------------------------------------------
int Cursor::staffIdx() const
{
return _track / VOICES;
}
//---------------------------------------------------------
// voice
//---------------------------------------------------------
int Cursor::voice() const
{
return _track % VOICES;
}
//---------------------------------------------------------
// nextInTrack
// go to first segment at or after _segment which has notes / rests in _track
//---------------------------------------------------------
inline void Cursor::firstChordRestInTrack()
{
2012-07-26 16:07:23 +02:00
while (_segment && _segment->element(_track) == 0)
2013-06-12 14:23:57 +02:00
_segment = _segment->next1(Segment::SegChordRest);
}
2013-05-13 18:49:17 +02:00
}