You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
rosegarden/src/base/Segment.cpp

1293 lines
31 KiB

/*
Rosegarden
A sequencer and musical notation editor.
This program is Copyright 2000-2008
Guillaume Laurent <glaurent@telegraph-road.org>,
Chris Cannam <cannam@all-day-breakfast.com>,
Richard Bown <bownie@bownie.com>
The moral right of the authors to claim authorship of this work
has been asserted.
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License as
published by the Free Software Foundation; either version 2 of the
License, or (at your option) any later version. See the file
COPYING included with this distribution for more information.
*/
#include "Segment.h"
#include "NotationTypes.h"
#include "BaseProperties.h"
#include "Composition.h"
#include "BasicQuantizer.h"
#include "Profiler.h"
#include <iostream>
#include <algorithm>
#include <iterator>
#include <cstdio>
#include <typeinfo>
namespace Rosegarden
{
using std::cerr;
using std::endl;
using std::string;
//#define DEBUG_NORMALIZE_RESTS 1
static int _runtimeSegmentId = 0;
Segment::Segment(SegmentType segmentType, timeT startTime) :
std::multiset<Event*, Event::EventCmp>(),
m_composition(0),
m_startTime(startTime),
m_endMarkerTime(0),
m_endTime(startTime),
m_track(0),
m_type(segmentType),
m_colourIndex(0),
m_id(0),
m_audioFileId(0),
m_unstretchedFileId(0),
m_stretchRatio(1.0),
m_audioStartTime(0, 0),
m_audioEndTime(0, 0),
m_repeating(false),
m_quantizer(new BasicQuantizer()),
m_quantize(false),
m_transpose(0),
m_delay(0),
m_realTimeDelay(0, 0),
m_clefKeyList(0),
m_runtimeSegmentId(_runtimeSegmentId++),
m_snapGridSize(-1),
m_viewFeatures(0),
m_autoFade(false),
m_fadeInTime(Rosegarden::RealTime::zeroTime),
m_fadeOutTime(Rosegarden::RealTime::zeroTime),
m_highestPlayable(127),
m_lowestPlayable(0)
{
}
Segment::Segment(const Segment &segment):
std::multiset<Event*, Event::EventCmp>(),
m_composition(0), // Composition should decide what's in it and what's not
m_startTime(segment.getStartTime()),
m_endMarkerTime(segment.m_endMarkerTime ?
new timeT(*segment.m_endMarkerTime) : 0),
m_endTime(segment.getEndTime()),
m_track(segment.getTrack()),
m_type(segment.getType()),
m_label(segment.getLabel()),
m_colourIndex(segment.getColourIndex()),
m_id(0),
m_audioFileId(segment.getAudioFileId()),
m_unstretchedFileId(segment.getUnstretchedFileId()),
m_stretchRatio(segment.getStretchRatio()),
m_audioStartTime(segment.getAudioStartTime()),
m_audioEndTime(segment.getAudioEndTime()),
m_repeating(segment.isRepeating()),
m_quantizer(new BasicQuantizer(segment.m_quantizer->getUnit(),
segment.m_quantizer->getDoDurations())),
m_quantize(segment.hasQuantization()),
m_transpose(segment.getTranspose()),
m_delay(segment.getDelay()),
m_realTimeDelay(segment.getRealTimeDelay()),
m_clefKeyList(0),
m_runtimeSegmentId(_runtimeSegmentId++),
m_snapGridSize(-1),
m_viewFeatures(0),
m_autoFade(segment.isAutoFading()),
m_fadeInTime(segment.getFadeInTime()),
m_fadeOutTime(segment.getFadeOutTime()),
m_highestPlayable(127),
m_lowestPlayable(0)
{
for (const_iterator it = segment.begin();
segment.isBeforeEndMarker(it); ++it) {
insert(new Event(**it));
}
}
Segment::~Segment()
{
if (!m_observers.empty()) {
cerr << "Warning: Segment::~Segment() with " << m_observers.size()
<< " observers still extant" << endl;
cerr << "Observers are:";
for (ObserverSet::const_iterator i = m_observers.begin();
i != m_observers.end(); ++i) {
cerr << " " << (void *)(*i);
cerr << " [" << typeid(**i).name() << "]";
}
cerr << endl;
}
notifySourceDeletion();
if (m_composition) m_composition->detachSegment(this);
if (m_clefKeyList) {
// don't delete contents of m_clefKeyList: the pointers
// are just aliases for events in the main segment
delete m_clefKeyList;
}
// Clear EventRulers
//
EventRulerListIterator it;
for (it = m_eventRulerList.begin(); it != m_eventRulerList.end(); ++it) delete *it;
m_eventRulerList.clear();
// delete content
for (iterator it = begin(); it != end(); ++it) delete (*it);
delete m_endMarkerTime;
}
void
Segment::setTrack(TrackId id)
{
Composition *c = m_composition;
if (c) c->weakDetachSegment(this); // sets m_composition to 0
TrackId oldTrack = m_track;
m_track = id;
if (c) {
c->weakAddSegment(this);
c->updateRefreshStatuses();
c->notifySegmentTrackChanged(this, oldTrack, id);
}
}
timeT
Segment::getStartTime() const
{
return m_startTime;
}
timeT
Segment::getEndMarkerTime() const
{
timeT endTime;
if (m_type == Audio && m_composition) {
RealTime startRT = m_composition->getElapsedRealTime(m_startTime);
RealTime endRT = startRT - m_audioStartTime + m_audioEndTime;
endTime = m_composition->getElapsedTimeForRealTime(endRT);
} else {
if (m_endMarkerTime) {
endTime = *m_endMarkerTime;
} else {
endTime = getEndTime();
}
if (m_composition) {
endTime = std::min(endTime, m_composition->getEndMarker());
}
}
return endTime;
}
timeT
Segment::getEndTime() const
{
if (m_type == Audio && m_composition) {
RealTime startRT = m_composition->getElapsedRealTime(m_startTime);
RealTime endRT = startRT - m_audioStartTime + m_audioEndTime;
return m_composition->getElapsedTimeForRealTime(endRT);
} else {
return m_endTime;
}
}
void
Segment::setStartTime(timeT t)
{
int dt = t - m_startTime;
if (dt == 0) return;
// reset the time of all events. can't just setAbsoluteTime on these,
// partly 'cos we're not allowed, partly 'cos it might screw up the
// quantizer (which is why we're not allowed)
// still, this is rather unsatisfactory
FastVector<Event *> events;
for (iterator i = begin(); i != end(); ++i) {
events.push_back((*i)->copyMoving(dt));
}
timeT previousEndTime = m_endTime;
erase(begin(), end());
m_endTime = previousEndTime + dt;
if (m_endMarkerTime) *m_endMarkerTime += dt;
if (m_composition) m_composition->setSegmentStartTime(this, t);
else m_startTime = t;
for (int i = 0; i < events.size(); ++i) {
insert(events[i]);
}
notifyStartChanged(m_startTime);
updateRefreshStatuses(m_startTime, m_endTime);
}
void
Segment::setEndMarkerTime(timeT t)
{
if (t < m_startTime) t = m_startTime;
if (m_type == Audio) {
if (m_endMarkerTime) *m_endMarkerTime = t;
else m_endMarkerTime = new timeT(t);
RealTime oldAudioEndTime = m_audioEndTime;
if (m_composition) {
m_audioEndTime = m_audioStartTime +
m_composition->getRealTimeDifference(m_startTime, t);
if (oldAudioEndTime != m_audioEndTime) {
notifyEndMarkerChange(m_audioEndTime < oldAudioEndTime);
}
}
} else {
timeT endTime = getEndTime();
timeT oldEndMarker = getEndMarkerTime();
bool shorten = (t < oldEndMarker);
if (t > endTime) {
fillWithRests(endTime, t);
if (oldEndMarker < endTime) {
updateRefreshStatuses(oldEndMarker, t);
}
} else {
// only need to do this if we aren't inserting or
// deleting any actual events
if (oldEndMarker < t) {
updateRefreshStatuses(oldEndMarker, t);
}
updateRefreshStatuses(t, endTime);
}
if (m_endMarkerTime) *m_endMarkerTime = t;
else m_endMarkerTime = new timeT(t);
notifyEndMarkerChange(shorten);
}
}
void
Segment::setEndTime(timeT t)
{
timeT endTime = getEndTime();
if (t < m_startTime) t = m_startTime;
if (m_type == Audio) {
setEndMarkerTime(t);
} else {
if (t < endTime) {
erase(findTime(t), end());
endTime = getEndTime();
if (m_endMarkerTime && endTime < *m_endMarkerTime) {
*m_endMarkerTime = endTime;
notifyEndMarkerChange(true);
}
} else if (t > endTime) {
fillWithRests(endTime, t);
}
}
}
Segment::iterator
Segment::getEndMarker()
{
if (m_endMarkerTime) {
return findTime(*m_endMarkerTime);
} else {
return end();
}
}
bool
Segment::isBeforeEndMarker(const_iterator i) const
{
if (i == end()) return false;
timeT absTime = (*i)->getAbsoluteTime();
timeT endTime = getEndMarkerTime();
return ((absTime < endTime) ||
(absTime == endTime && (*i)->getDuration() == 0));
}
void
Segment::clearEndMarker()
{
delete m_endMarkerTime;
m_endMarkerTime = 0;
notifyEndMarkerChange(false);
}
const timeT *
Segment::getRawEndMarkerTime() const
{
return m_endMarkerTime;
}
void
Segment::updateRefreshStatuses(timeT startTime, timeT endTime)
{
for(unsigned int i = 0; i < m_refreshStatusArray.size(); ++i)
m_refreshStatusArray.getRefreshStatus(i).push(startTime, endTime);
}
Segment::iterator
Segment::insert(Event *e)
{
assert(e);
timeT t0 = e->getAbsoluteTime();
timeT t1 = t0 + e->getDuration();
if (t0 < m_startTime ||
(begin() == end() && t0 > m_startTime)) {
if (m_composition) m_composition->setSegmentStartTime(this, t0);
else m_startTime = t0;
notifyStartChanged(m_startTime);
}
if (t1 > m_endTime ||
begin() == end()) {
timeT oldTime = m_endTime;
m_endTime = t1;
notifyEndMarkerChange(m_endTime < oldTime);
}
iterator i = std::multiset<Event*, Event::EventCmp>::insert(e);
notifyAdd(e);
updateRefreshStatuses(e->getAbsoluteTime(),
e->getAbsoluteTime() + e->getDuration());
return i;
}
void
Segment::updateEndTime()
{
m_endTime = m_startTime;
for (iterator i = begin(); i != end(); ++i) {
timeT t = (*i)->getAbsoluteTime() + (*i)->getDuration();
if (t > m_endTime) m_endTime = t;
}
}
void
Segment::erase(iterator pos)
{
Event *e = *pos;
assert(e);
timeT t0 = e->getAbsoluteTime();
timeT t1 = t0 + e->getDuration();
std::multiset<Event*, Event::EventCmp>::erase(pos);
notifyRemove(e);
delete e;
updateRefreshStatuses(t0, t1);
if (t0 == m_startTime && begin() != end()) {
timeT startTime = (*begin())->getAbsoluteTime();
if (m_composition) m_composition->setSegmentStartTime(this, startTime);
else m_startTime = startTime;
notifyStartChanged(m_startTime);
}
if (t1 == m_endTime) {
updateEndTime();
}
}
void
Segment::erase(iterator from, iterator to)
{
timeT startTime = 0, endTime = m_endTime;
if (from != end()) startTime = (*from)->getAbsoluteTime();
if (to != end()) endTime = (*to)->getAbsoluteTime() + (*to)->getDuration();
// Not very efficient, but without an observer event for
// multiple erase we can't do any better.
for (Segment::iterator i = from; i != to; ) {
Segment::iterator j(i);
++j;
Event *e = *i;
assert(e);
std::multiset<Event*, Event::EventCmp>::erase(i);
notifyRemove(e);
delete e;
i = j;
}
if (startTime == m_startTime && begin() != end()) {
timeT startTime = (*begin())->getAbsoluteTime();
if (m_composition) m_composition->setSegmentStartTime(this, startTime);
else m_startTime = startTime;
notifyStartChanged(m_startTime);
}
if (endTime == m_endTime) {
updateEndTime();
}
updateRefreshStatuses(startTime, endTime);
}
bool
Segment::eraseSingle(Event* e)
{
iterator elPos = findSingle(e);
if (elPos != end()) {
erase(elPos);
return true;
} else return false;
}
Segment::iterator
Segment::findSingle(Event* e)
{
iterator res = end();
std::pair<iterator, iterator> interval = equal_range(e);
for (iterator i = interval.first; i != interval.second; ++i) {
if (*i == e) {
res = i;
break;
}
}
return res;
}
Segment::iterator
Segment::findTime(timeT t)
{
Event dummy("dummy", t, 0, MIN_SUBORDERING);
return lower_bound(&dummy);
}
Segment::iterator
Segment::findNearestTime(timeT t)
{
iterator i = findTime(t);
if (i == end() || (*i)->getAbsoluteTime() > t) {
if (i == begin()) return end();
else --i;
}
return i;
}
timeT
Segment::getBarStartForTime(timeT t) const
{
if (t < getStartTime()) t = getStartTime();
return getComposition()->getBarStartForTime(t);
}
timeT
Segment::getBarEndForTime(timeT t) const
{
if (t > getEndMarkerTime()) t = getEndMarkerTime();
return getComposition()->getBarEndForTime(t);
}
int Segment::getNextId() const
{
return m_id++;
}
void
Segment::fillWithRests(timeT endTime)
{
fillWithRests(getEndTime(), endTime);
}
void
Segment::fillWithRests(timeT startTime, timeT endTime)
{
if (startTime < m_startTime) {
if (m_composition) m_composition->setSegmentStartTime(this, startTime);
else m_startTime = startTime;
notifyStartChanged(m_startTime);
}
TimeSignature ts;
timeT sigTime = 0;
if (getComposition()) {
sigTime = getComposition()->getTimeSignatureAt(startTime, ts);
}
timeT restDuration = endTime - startTime;
if (restDuration <= 0) return;
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "fillWithRests (" << startTime << "->" << endTime << "), composition "
<< (getComposition() ? "exists" : "does not exist") << ", sigTime "
<< sigTime << ", timeSig duration " << ts.getBarDuration() << ", restDuration " << restDuration << endl;
#endif
DurationList dl;
ts.getDurationListForInterval(dl, restDuration, startTime - sigTime);
timeT acc = startTime;
for (DurationList::iterator i = dl.begin(); i != dl.end(); ++i) {
Event *e = new Event(Note::EventRestType, acc, *i,
Note::EventRestSubOrdering);
insert(e);
acc += *i;
}
}
void
Segment::normalizeRests(timeT startTime, timeT endTime)
{
Profiler profiler("Segment::normalizeRests");
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests (" << startTime << "->" << endTime << "), segment starts at " << m_startTime << endl;
#endif
if (startTime < m_startTime) {
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests: pulling start time back from "
<< m_startTime << " to " << startTime << endl;
#endif
if (m_composition) m_composition->setSegmentStartTime(this, startTime);
else m_startTime = startTime;
notifyStartChanged(m_startTime);
}
//!!! Need to remove the rests then relocate the start time
// and get the notation end time for the nearest note before that
// (?)
//!!! We need to insert rests at fictitious unquantized times that
//are broadly correct, so as to maintain ordering of notes and
//rests in the unquantized segment. The quantized times should go
//in notation-prefix properties.
// Preliminary: If there are any time signature changes between
// the start and end times, consider separately each of the sections
// they divide the range up into.
Composition *composition = getComposition();
if (composition) {
int timeSigNo = composition->getTimeSignatureNumberAt(startTime);
if (timeSigNo < composition->getTimeSignatureCount() - 1) {
timeT nextSigTime =
composition->getTimeSignatureChange(timeSigNo + 1).first;
if (nextSigTime < endTime) {
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests: divide-and-conquer on timesig at " << nextSigTime << endl;
#endif
normalizeRests(startTime, nextSigTime);
normalizeRests(nextSigTime, endTime);
return;
}
}
}
// First stage: erase all existing non-tupleted rests in this range.
timeT segmentEndTime = m_endTime;
iterator ia = findNearestTime(startTime);
if (ia == end()) ia = begin();
if (ia == end()) { // the segment is empty
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests: empty segment" << endl;
#endif
fillWithRests(startTime, endTime);
return;
} else {
if (startTime > (*ia)->getNotationAbsoluteTime()) {
startTime = (*ia)->getNotationAbsoluteTime();
}
}
iterator ib = findTime(endTime);
if (ib == end()) {
if (ib != begin()) {
--ib;
// if we're pointing at the real-end-time of the last event,
// use its notation-end-time instead
if (endTime == (*ib)->getAbsoluteTime() + (*ib)->getDuration()) {
endTime =
(*ib)->getNotationAbsoluteTime() +
(*ib)->getNotationDuration();
}
++ib;
}
} else {
endTime = (*ib)->getNotationAbsoluteTime();
}
// If there's a rest preceding the start time, with no notes
// between us and it, and if it doesn't have precisely the
// right duration, then we need to normalize it too
//!!! needs modification for new scheme
iterator scooter = ia;
while (scooter-- != begin()) {
// if ((*scooter)->isa(Note::EventRestType)) { //!!! experimental
if ((*scooter)->getDuration() > 0) {
if ((*scooter)->getNotationAbsoluteTime() +
(*scooter)->getNotationDuration() !=
startTime) {
startTime = (*scooter)->getNotationAbsoluteTime();
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests: scooting back to " << startTime << endl;
#endif
ia = scooter;
}
break;
/*!!!
} else if ((*scooter)->getDuration() > 0) {
break;
*/
}
}
for (iterator i = ia, j = i; i != ib && i != end(); i = j) {
++j;
if ((*i)->isa(Note::EventRestType) &&
!(*i)->has(BaseProperties::BEAMED_GROUP_TUPLET_BASE)) {
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests: erasing rest at " << (*i)->getAbsoluteTime() << endl;
#endif
erase(i);
}
}
// It's possible we've just removed all the events between here
// and the end of the segment, if they were all rests. Check.
if (endTime < segmentEndTime && m_endTime < segmentEndTime) {
endTime = segmentEndTime;
}
// Second stage: find the gaps that need to be filled with
// rests. We don't mind about the case where two simultaneous
// notes end at different times -- we're only interested in
// the one ending sooner. Each time an event ends, we start
// a candidate gap.
std::vector<std::pair<timeT, timeT> > gaps;
timeT lastNoteStarts = startTime;
timeT lastNoteEnds = startTime;
// Re-find this, as it might have been erased
ia = findNearestTime(startTime);
if (ia == end()) {
// already have good lastNoteStarts, lastNoteEnds
ia = begin();
} else {
lastNoteStarts = (*ia)->getNotationAbsoluteTime();
lastNoteEnds = lastNoteStarts;
}
if (ib != end()) {
//!!! This and related code really need to get a quantized
// absolute time of a note event that has the same unquantized
// time as ib, not necessarily of ib itself... or else the
// quantizer needs to set the quantized times of all non-note
// events that happen at the same unquantized time as a note
// event to the same as that of the note event... yeah, that's
// probably the right thing
endTime = (*ib)->getNotationAbsoluteTime();
// was this just a nasty hack?
++ib;
}
iterator i = ia;
for (; i != ib && i != end(); ++i) {
// Boundary events for sets of rests may be notes (obviously),
// text events (because they need to be "attached" to
// something that has the correct timing), or rests (any
// remaining rests in this area have tuplet data so should be
// treated as "hard" rests);
if (!((*i)->isa(Note::EventType) ||
(*i)->isa(Text::EventType) ||
(*i)->isa(Note::EventRestType))) {
continue;
}
timeT thisNoteStarts = (*i)->getNotationAbsoluteTime();
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests: scanning: thisNoteStarts " << thisNoteStarts
<< ", lastNoteStarts " << lastNoteStarts
<< ", lastNoteEnds " << lastNoteEnds << endl;
#endif
/* BR #988185: "Notation: Rest can be simultaneous with note but follow it"
This conditional tested whether a note started before the
preceding note ended, and if so inserted rests simultaneous
with the preceding note to make up the gap. Without the
ability to lay out those rests partwise, this is never any
better than plain confusing. Revert the change.
if (thisNoteStarts < lastNoteEnds &&
thisNoteStarts > lastNoteStarts) {
gaps.push_back(std::pair<timeT, timeT>
(lastNoteStarts,
thisNoteStarts - lastNoteStarts));
}
*/
if (thisNoteStarts > lastNoteEnds) {
gaps.push_back(std::pair<timeT, timeT>
(lastNoteEnds,
thisNoteStarts - lastNoteEnds));
}
lastNoteStarts = thisNoteStarts;
lastNoteEnds = thisNoteStarts + (*i)->getNotationDuration();
}
if (endTime > lastNoteEnds) {
gaps.push_back(std::pair<timeT, timeT>
(lastNoteEnds, endTime - lastNoteEnds));
}
timeT duration;
for (unsigned int gi = 0; gi < gaps.size(); ++gi) {
#ifdef DEBUG_NORMALIZE_RESTS
cerr << "normalizeRests: gap " << gi << ": " << gaps[gi].first << " -> " << (gaps[gi].first + gaps[gi].second) << endl;
#endif
startTime = gaps[gi].first;
duration = gaps[gi].second;
if (duration >= Note(Note::Shortest).getDuration()) {
fillWithRests(startTime, startTime + duration);
}
}
}
void Segment::getTimeSlice(timeT absoluteTime, iterator &start, iterator &end)
{
Event dummy("dummy", absoluteTime, 0, MIN_SUBORDERING);
// No, this won't work -- we need to include things that don't
// compare equal because they have different suborderings, as long
// as they have the same times
// std::pair<iterator, iterator> res = equal_range(&dummy);
// start = res.first;
// end = res.second;
// Got to do this instead:
start = end = lower_bound(&dummy);
while (end != this->end() &&
(*end)->getAbsoluteTime() == (*start)->getAbsoluteTime())
++end;
}
void Segment::getTimeSlice(timeT absoluteTime, const_iterator &start, const_iterator &end)
const
{
Event dummy("dummy", absoluteTime, 0, MIN_SUBORDERING);
start = end = lower_bound(&dummy);
while (end != this->end() &&
(*end)->getAbsoluteTime() == (*start)->getAbsoluteTime())
++end;
}
void
Segment::setQuantization(bool quantize)
{
if (m_quantize != quantize) {
m_quantize = quantize;
if (m_quantize) {
m_quantizer->quantize(this, begin(), end());
} else {
m_quantizer->unquantize(this, begin(), end());
}
}
}
bool
Segment::hasQuantization() const
{
return m_quantize;
}
void
Segment::setQuantizeLevel(timeT unit)
{
if (m_quantizer->getUnit() == unit) return;
m_quantizer->setUnit(unit);
if (m_quantize) m_quantizer->quantize(this, begin(), end());
}
const BasicQuantizer *
Segment::getQuantizer() const
{
return m_quantizer;
}
void
Segment::setRepeating(bool value)
{
m_repeating = value;
if (m_composition) {
m_composition->updateRefreshStatuses();
m_composition->notifySegmentRepeatChanged(this, value);
}
}
void
Segment::setDelay(timeT delay)
{
m_delay = delay;
if (m_composition) {
// don't updateRefreshStatuses() - affects playback only
m_composition->notifySegmentEventsTimingChanged(this, delay, RealTime::zeroTime);
}
}
void
Segment::setRealTimeDelay(RealTime delay)
{
m_realTimeDelay = delay;
if (m_composition) {
// don't updateRefreshStatuses() - affects playback only
m_composition->notifySegmentEventsTimingChanged(this, 0, delay);
}
}
void
Segment::setTranspose(int transpose)
{
m_transpose = transpose;
if (m_composition) {
// don't updateRefreshStatuses() - affects playback only
m_composition->notifySegmentTransposeChanged(this, transpose);
}
}
void
Segment::setAudioFileId(unsigned int id)
{
m_audioFileId = id;
updateRefreshStatuses(getStartTime(), getEndTime());
}
void
Segment::setUnstretchedFileId(unsigned int id)
{
m_unstretchedFileId = id;
}
void
Segment::setStretchRatio(float ratio)
{
m_stretchRatio = ratio;
}
void
Segment::setAudioStartTime(const RealTime &time)
{
m_audioStartTime = time;
updateRefreshStatuses(getStartTime(), getEndTime());
}
void
Segment::setAudioEndTime(const RealTime &time)
{
RealTime oldAudioEndTime = m_audioEndTime;
m_audioEndTime = time;
updateRefreshStatuses(getStartTime(), getEndTime());
notifyEndMarkerChange(time < oldAudioEndTime);
}
void
Segment::setAutoFade(bool value)
{
m_autoFade = value;
updateRefreshStatuses(getStartTime(), getEndTime());
}
void
Segment::setFadeInTime(const RealTime &time)
{
m_fadeInTime = time;
updateRefreshStatuses(getStartTime(), getEndTime());
}
void
Segment::setFadeOutTime(const RealTime &time)
{
m_fadeOutTime = time;
updateRefreshStatuses(getStartTime(), getEndTime());
}
void
Segment::setLabel(const std::string &label)
{
m_label = label;
if (m_composition) m_composition->updateRefreshStatuses();
notifyAppearanceChange();
}
bool
Segment::ClefKeyCmp::operator()(const Event *e1, const Event *e2) const
{
if (e1->getType() == e2->getType()) return Event::EventCmp()(e1, e2);
else return e1->getType() < e2->getType();
}
Clef
Segment::getClefAtTime(timeT time) const
{
timeT ctime;
return getClefAtTime(time, ctime);
}
Clef
Segment::getClefAtTime(timeT time, timeT &ctime) const
{
if (!m_clefKeyList) return Clef();
Event ec(Clef::EventType, time);
ClefKeyList::iterator i = m_clefKeyList->lower_bound(&ec);
while (i == m_clefKeyList->end() ||
(*i)->getAbsoluteTime() > time ||
(*i)->getType() != Clef::EventType) {
if (i == m_clefKeyList->begin()) {
ctime = getStartTime();
return Clef();
}
--i;
}
try {
ctime = (*i)->getAbsoluteTime();
return Clef(**i);
} catch (const Exception &e) {
std::cerr << "Segment::getClefAtTime(" << time
<< "): bogus clef in ClefKeyList: event dump follows:"
<< std::endl;
(*i)->dump(std::cerr);
return Clef();
}
}
Key
Segment::getKeyAtTime(timeT time) const
{
timeT ktime;
return getKeyAtTime(time, ktime);
}
Key
Segment::getKeyAtTime(timeT time, timeT &ktime) const
{
if (!m_clefKeyList) return Key();
Event ek(Key::EventType, time);
ClefKeyList::iterator i = m_clefKeyList->lower_bound(&ek);
while (i == m_clefKeyList->end() ||
(*i)->getAbsoluteTime() > time ||
(*i)->getType() != Key::EventType) {
if (i == m_clefKeyList->begin()) {
ktime = getStartTime();
return Key();
}
--i;
}
try {
ktime = (*i)->getAbsoluteTime();
return Key(**i);
} catch (const Exception &e) {
std::cerr << "Segment::getClefAtTime(" << time
<< "): bogus key in ClefKeyList: event dump follows:"
<< std::endl;
(*i)->dump(std::cerr);
return Key();
}
}
void
Segment::getFirstClefAndKey(Clef &clef, Key &key)
{
bool keyFound = false;
bool clefFound = false;
clef = Clef(); // Default clef
key = Key(); // Default key signature
iterator i = begin();
while (i!=end()) {
// Keep current clef and key as soon as a note or rest event is found
if ((*i)->isa(Note::EventRestType) || (*i)->isa(Note::EventType)) return;
// Remember the first clef event found
if ((*i)->isa(Clef::EventType)) {
clef = Clef(*(*i));
// and return if a key has already been found
if (keyFound) return;
clefFound = true;
}
// Remember the first key event found
if ((*i)->isa(Key::EventType)) {
key = Key(*(*i));
// and return if a clef has already been found
if (clefFound) return;
keyFound = true;
}
++i;
}
}
timeT
Segment::getRepeatEndTime() const
{
timeT endMarker = getEndMarkerTime();
if (m_repeating && m_composition) {
Composition::iterator i(m_composition->findSegment(this));
assert(i != m_composition->end());
++i;
if (i != m_composition->end() && (*i)->getTrack() == getTrack()) {
timeT t = (*i)->getStartTime();
if (t < endMarker) return endMarker;
else return t;
} else {
return m_composition->getEndMarker();
}
}
return endMarker;
}
void
Segment::notifyAdd(Event *e) const
{
if (e->isa(Clef::EventType) || e->isa(Key::EventType)) {
if (!m_clefKeyList) m_clefKeyList = new ClefKeyList;
m_clefKeyList->insert(e);
}
for (ObserverSet::const_iterator i = m_observers.begin();
i != m_observers.end(); ++i) {
(*i)->eventAdded(this, e);
}
}
void
Segment::notifyRemove(Event *e) const
{
if (m_clefKeyList && (e->isa(Clef::EventType) || e->isa(Key::EventType))) {
ClefKeyList::iterator i;
for (i = m_clefKeyList->find(e); i != m_clefKeyList->end(); ++i) {
// fix for bug#1485643 (crash erasing a duplicated key signature)
if ((*i) == e) {
m_clefKeyList->erase(i);
break;
}
}
}
for (ObserverSet::const_iterator i = m_observers.begin();
i != m_observers.end(); ++i) {
(*i)->eventRemoved(this, e);
}
}
void
Segment::notifyAppearanceChange() const
{
for (ObserverSet::const_iterator i = m_observers.begin();
i != m_observers.end(); ++i) {
(*i)->appearanceChanged(this);
}
}
void
Segment::notifyStartChanged(timeT newTime)
{
for (ObserverSet::const_iterator i = m_observers.begin();
i != m_observers.end(); ++i) {
(*i)->startChanged(this, newTime);
}
if (m_composition) {
m_composition->notifySegmentStartChanged(this, newTime);
}
}
void
Segment::notifyEndMarkerChange(bool shorten)
{
for (ObserverSet::const_iterator i = m_observers.begin();
i != m_observers.end(); ++i) {
(*i)->endMarkerTimeChanged(this, shorten);
}
if (m_composition) {
m_composition->notifySegmentEndMarkerChange(this, shorten);
}
}
void
Segment::notifySourceDeletion() const
{
for (ObserverSet::const_iterator i = m_observers.begin();
i != m_observers.end(); ++i) {
(*i)->segmentDeleted(this);
}
}
void
Segment::setColourIndex(const unsigned int input)
{
m_colourIndex = input;
updateRefreshStatuses(getStartTime(), getEndTime());
if (m_composition) m_composition->updateRefreshStatuses();
notifyAppearanceChange();
}
void
Segment::addEventRuler(const std::string &type, int controllerValue, bool active)
{
EventRulerListConstIterator it;
for (it = m_eventRulerList.begin(); it != m_eventRulerList.end(); ++it)
if ((*it)->m_type == type && (*it)->m_controllerValue == controllerValue)
return;
m_eventRulerList.push_back(new EventRuler(type, controllerValue, active));
}
bool
Segment::deleteEventRuler(const std::string &type, int controllerValue)
{
EventRulerListIterator it;
for (it = m_eventRulerList.begin(); it != m_eventRulerList.end(); ++it)
{
if ((*it)->m_type == type && (*it)->m_controllerValue == controllerValue)
{
delete *it;
m_eventRulerList.erase(it);
return true;
}
}
return false;
}
Segment::EventRuler*
Segment::getEventRuler(const std::string &type, int controllerValue)
{
EventRulerListConstIterator it;
for (it = m_eventRulerList.begin(); it != m_eventRulerList.end(); ++it)
if ((*it)->m_type == type && (*it)->m_controllerValue == controllerValue)
return *it;
return 0;
}
SegmentHelper::~SegmentHelper() { }
void
SegmentRefreshStatus::push(timeT from, timeT to)
{
if (!needsRefresh()) { // don't do anything subtle - just erase the old data
m_from = from;
m_to = to;
} else { // accumulate on what was already there
if (from < m_from) m_from = from;
if (to > m_to) m_to = to;
}
if (m_to < m_from) std::swap(m_from, m_to);
setNeedsRefresh(true);
}
}