summaryrefslogtreecommitdiffstats
path: root/src/base/Sets.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/base/Sets.h')
-rw-r--r--src/base/Sets.h698
1 files changed, 698 insertions, 0 deletions
diff --git a/src/base/Sets.h b/src/base/Sets.h
new file mode 100644
index 0000000..4fe14d1
--- /dev/null
+++ b/src/base/Sets.h
@@ -0,0 +1,698 @@
+// -*- c-basic-offset: 4 -*-
+
+/*
+ Rosegarden
+ A sequencer and musical notation editor.
+
+ This program is Copyright 2000-2008
+ Guillaume Laurent <[email protected]>,
+ Chris Cannam <[email protected]>,
+ Richard Bown <[email protected]>
+
+ 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.
+*/
+
+#ifndef _SETS_H_
+#define _SETS_H_
+
+#include <vector>
+#include <algorithm>
+
+#include "Event.h"
+#include "Segment.h"
+#include "CompositionTimeSliceAdapter.h"
+#include "BaseProperties.h"
+#include "NotationTypes.h"
+#include "MidiTypes.h"
+#include "Quantizer.h"
+
+namespace Rosegarden
+{
+
+class Quantizer;
+
+/**
+ * A "set" in Rosegarden terminology is a collection of elements found
+ * in a container (indeed, a subset of that container) all of which
+ * share a particular property and are located near to one another:
+ * generally either contiguous or within the same bar. The elements
+ * are most usually Events and the container most usually a Segment,
+ * and although this does not have to be the case (for other examples
+ * see gui/notationsets.h), the elements do have to be convertible to
+ * Events somehow.
+ *
+ * To construct a set requires (at least) a container reference plus
+ * an iterator into that container. The constructor (or more
+ * precisely the initialise() method called by the constructor) then
+ * scans the surrounding area of the list for the maximal set of
+ * contiguous or within-the-same-bar elements before and after the
+ * passed-in iterator such that all elements are in the same set
+ * (i.e. Chord, BeamedGroup etc) as the one that the passed-in
+ * iterator pointed to.
+ *
+ * The extents of the set within the list can then be discovered via
+ * getInitialElement() and getFinalElement(). If the iterator passed
+ * in to the constructor was at end() or did not point to an element
+ * that could be a member of this kind of set, getInitialElement()
+ * will return end(); if the passed-in iterator pointed to the only
+ * member of this set, getInitialElement() and getFinalElement() will
+ * be equal.
+ *
+ * These classes are not intended to be stored anywhere; all they
+ * contain is iterators into the main container, and those might not
+ * persist. Instead you should create these on-the-fly when you want,
+ * for example, to consider a note as part of a chord; and then you
+ * should let them expire when you've finished with them.
+ */
+
+template <class Element, class Container>
+class AbstractSet // abstract base
+{
+public:
+ typedef typename Container::iterator Iterator;
+
+ virtual ~AbstractSet() { }
+
+ /**
+ * getInitialElement() returns end() if there are no elements in
+ * the set. getInitialElement() == getFinalElement() if there is
+ * only one element in the set
+ */
+ Iterator getInitialElement() const { return m_initial; }
+ Iterator getFinalElement() const { return m_final; }
+
+ /// only return note elements; will return end() if there are none
+ Iterator getInitialNote() const { return m_initialNote; }
+ Iterator getFinalNote() const { return m_finalNote; }
+
+ /**
+ * only elements with duration > 0 are candidates for shortest and
+ * longest; these will return end() if there are no such elements
+ */
+ Iterator getLongestElement() const { return m_longest; }
+ Iterator getShortestElement() const { return m_shortest; }
+
+ /// these will return end() if there are no note elements in the set
+ Iterator getHighestNote() const { return m_highest; }
+ Iterator getLowestNote() const { return m_lowest; }
+
+ virtual bool contains(const Iterator &) const = 0;
+
+ /// Return the pointed-to element, in Event form (public to work around gcc-2.95 bug)
+ static Event *getAsEvent(const Iterator &i);
+
+protected:
+ AbstractSet(Container &c, Iterator elementInSet, const Quantizer *);
+ void initialise();
+
+ /// Return true if this element is not definitely beyond bounds of set
+ virtual bool test(const Iterator &i) = 0;
+
+ /// Return true if this element, known to test() true, is a set member
+ virtual bool sample(const Iterator &i, bool goingForwards);
+
+ Container &getContainer() const { return m_container; }
+ const Quantizer &getQuantizer() const { return *m_quantizer; }
+
+ // Data members:
+
+ Container &m_container;
+ Iterator m_initial, m_final, m_initialNote, m_finalNote;
+ Iterator m_shortest, m_longest, m_highest, m_lowest;
+ Iterator m_baseIterator;
+ const Quantizer *m_quantizer;
+};
+
+
+/**
+ * Chord is subclassed from a vector of iterators; this vector
+ * contains iterators pointing at all the notes in the chord, in
+ * ascending order of pitch. You can also track through all the
+ * events in the chord by iterating from getInitialElement() to
+ * getFinalElement(), but this will only get them in the order in
+ * which they appear in the original container.
+ *
+ * However, the notes in a chord might not be contiguous events in the
+ * container, as there could be other zero-duration events such as
+ * controllers (or even conceivably some short rests) between notes in
+ * the same chord, depending on the quantization settings. The Chord
+ * itself only contains iterators pointing at the notes, so if you
+ * want to iterate through all events spanned by the Chord, iterate
+ * from getInitialElement() to getFinalElement() instead.
+ *
+ * This class can tell you various things about the chord it
+ * describes, but not everything. It can't tell you whether the
+ * chord has a stem, for example, because that depends on the
+ * notation style and system in use. See gui/notationsets.h
+ * for a NotationChord class (subclassed from this) that can.
+ */
+
+template <class Element, class Container, bool singleStaff>
+class GenericChord : public AbstractSet<Element, Container>,
+ public std::vector<typename Container::iterator>
+{
+public:
+ typedef typename Container::iterator Iterator;
+
+ /* You only need to provide the clef and key if the notes
+ making up your chord lack HEIGHT_ON_STAFF properties, in
+ which case this constructor will write those properties
+ in to the chord for you */
+ GenericChord(Container &c,
+ Iterator elementInChord,
+ const Quantizer *quantizer,
+ PropertyName stemUpProperty = PropertyName::EmptyPropertyName);
+
+ virtual ~GenericChord();
+
+ virtual int getMarkCountForChord() const;
+ virtual std::vector<Mark> getMarksForChord() const;
+ virtual std::vector<int> getPitches() const;
+ virtual bool contains(const Iterator &) const;
+
+ /**
+ * Return an iterator pointing to the previous note before this
+ * chord, or container's end() if there is no previous note.
+ */
+ virtual Iterator getPreviousNote();
+
+ /**
+ * Return an iterator pointing to the next note after this chord,
+ * or container's end() if there is no next note. Remember this
+ * class can't know about Segment end marker times, so if your
+ * container is a Segment, check the returned note is actually
+ * before the end marker.
+ */
+ virtual Iterator getNextNote();
+
+ /**
+ * It's possible for a chord to surround (in the segment) elements
+ * that are not members of the chord. This function returns an
+ * iterator pointing to the first of those after the iterator that
+ * was passed to the chord's constructor. If there are none, it
+ * returns the container's end().
+ */
+ virtual Iterator getFirstElementNotInChord();
+
+ virtual int getSubOrdering() { return m_subordering; }
+
+protected:
+ virtual bool test(const Iterator&);
+ virtual bool sample(const Iterator&, bool goingForwards);
+
+ class PitchGreater {
+ public:
+ bool operator()(const Iterator &a, const Iterator &b);
+ };
+
+ void copyGroupProperties(Event *e0, Event *e1) const;
+
+ //--------------- Data members ---------------------------------
+
+ PropertyName m_stemUpProperty;
+ timeT m_time;
+ int m_subordering;
+ Iterator m_firstReject;
+};
+
+
+
+///
+/// Implementation only from here on.
+///
+
+// forward declare hack functions -- see Sets.C for an explanation
+
+extern long
+get__Int(Event *e, const PropertyName &name);
+
+extern bool
+get__Bool(Event *e, const PropertyName &name);
+
+extern std::string
+get__String(Event *e, const PropertyName &name);
+
+extern bool
+get__Int(Event *e, const PropertyName &name, long &ref);
+
+extern bool
+get__Bool(Event *e, const PropertyName &name, bool &ref);
+
+extern bool
+get__String(Event *e, const PropertyName &name, std::string &ref);
+
+extern bool
+isPersistent__Bool(Event *e, const PropertyName &name);
+
+extern void
+setMaybe__Int(Event *e, const PropertyName &name, long value);
+
+extern void
+setMaybe__String(Event *e, const PropertyName &name, const std::string &value);
+
+
+
+template <class Element, class Container>
+AbstractSet<Element, Container>::AbstractSet(Container &c,
+ Iterator i, const Quantizer *q):
+ m_container(c),
+ m_initial(c.end()),
+ m_final(c.end()),
+ m_initialNote(c.end()),
+ m_finalNote(c.end()),
+ m_shortest(c.end()),
+ m_longest(c.end()),
+ m_highest(c.end()),
+ m_lowest(c.end()),
+ m_baseIterator(i),
+ m_quantizer(q)
+{
+ // ...
+}
+
+template <class Element, class Container>
+void
+AbstractSet<Element, Container>::initialise()
+{
+ if (m_baseIterator == getContainer().end() || !test(m_baseIterator)) return;
+
+ m_initial = m_baseIterator;
+ m_final = m_baseIterator;
+ sample(m_baseIterator, true);
+
+ if (getAsEvent(m_baseIterator)->isa(Note::EventType)) {
+ m_initialNote = m_baseIterator;
+ m_finalNote = m_baseIterator;
+ }
+
+ Iterator i, j;
+
+ // first scan back to find an element not in the desired set,
+ // sampling everything as far back as the one after it
+
+ for (i = j = m_baseIterator; i != getContainer().begin() && test(--j); i = j){
+ if (sample(j, false)) {
+ m_initial = j;
+ if (getAsEvent(j)->isa(Note::EventType)) {
+ m_initialNote = j;
+ if (m_finalNote == getContainer().end()) {
+ m_finalNote = j;
+ }
+ }
+ }
+ }
+
+ j = m_baseIterator;
+
+ // then scan forwards to find an element not in the desired set,
+ // sampling everything as far forward as the one before it
+
+ for (i = j = m_baseIterator; ++j != getContainer().end() && test(j); i = j) {
+ if (sample(j, true)) {
+ m_final = j;
+ if (getAsEvent(j)->isa(Note::EventType)) {
+ m_finalNote = j;
+ if (m_initialNote == getContainer().end()) {
+ m_initialNote = j;
+ }
+ }
+ }
+ }
+}
+
+template <class Element, class Container>
+bool
+AbstractSet<Element, Container>::sample(const Iterator &i, bool)
+{
+ const Quantizer &q(getQuantizer());
+ Event *e = getAsEvent(i);
+ timeT d(q.getQuantizedDuration(e));
+
+ if (e->isa(Note::EventType) || d > 0) {
+ if (m_longest == getContainer().end() ||
+ d > q.getQuantizedDuration(getAsEvent(m_longest))) {
+// std::cerr << "New longest in set at duration " << d << " and time " << e->getAbsoluteTime() << std::endl;
+ m_longest = i;
+ }
+ if (m_shortest == getContainer().end() ||
+ d < q.getQuantizedDuration(getAsEvent(m_shortest))) {
+// std::cerr << "New shortest in set at duration " << d << " and time " << e->getAbsoluteTime() << std::endl;
+ m_shortest = i;
+ }
+ }
+
+ if (e->isa(Note::EventType)) {
+ long p = get__Int(e, BaseProperties::PITCH);
+
+ if (m_highest == getContainer().end() ||
+ p > get__Int(getAsEvent(m_highest), BaseProperties::PITCH)) {
+// std::cerr << "New highest in set at pitch " << p << " and time " << e->getAbsoluteTime() << std::endl;
+ m_highest = i;
+ }
+ if (m_lowest == getContainer().end() ||
+ p < get__Int(getAsEvent(m_lowest), BaseProperties::PITCH)) {
+// std::cerr << "New lowest in set at pitch " << p << " and time " << e->getAbsoluteTime() << std::endl;
+ m_lowest = i;
+ }
+ }
+
+ return true;
+}
+
+
+//////////////////////////////////////////////////////////////////////
+
+template <class Element, class Container, bool singleStaff>
+GenericChord<Element, Container, singleStaff>::GenericChord(Container &c,
+ Iterator i,
+ const Quantizer *q,
+ PropertyName stemUpProperty) :
+ AbstractSet<Element, Container>(c, i, q),
+ m_stemUpProperty(stemUpProperty),
+ m_time(q->getQuantizedAbsoluteTime(getAsEvent(i))),
+ m_subordering(getAsEvent(i)->getSubOrdering()),
+ m_firstReject(c.end())
+{
+ AbstractSet<Element, Container>::initialise();
+
+ if (std::vector<typename Container::iterator>::size() > 1) {
+ std::stable_sort(std::vector<typename Container::iterator>::begin(),
+ std::vector<typename Container::iterator>::end(),
+ PitchGreater());
+ }
+
+/*!!! this should all be removed ultimately
+// std::cerr << "GenericChord::GenericChord: pitches are:" << std::endl;
+ int prevPitch = -999;
+ for (unsigned int i = 0; i < size(); ++i) {
+ try {
+ int pitch = getAsEvent((*this)[i])->get<Int>(BaseProperties::PITCH);
+// std::cerr << i << ": " << pitch << std::endl;
+ if (pitch < prevPitch) {
+ cerr << "ERROR: Pitch less than previous pitch (" << pitch
+ << " < " << prevPitch << ")" << std::endl;
+ throw(1);
+ }
+ } catch (Event::NoData) {
+ std::cerr << i << ": no pitch property" << std::endl;
+ }
+ }
+*/
+}
+
+template <class Element, class Container, bool singleStaff>
+GenericChord<Element, Container, singleStaff>::~GenericChord()
+{
+}
+
+template <class Element, class Container, bool singleStaff>
+bool
+GenericChord<Element, Container, singleStaff>::test(const Iterator &i)
+{
+ Event *e = getAsEvent(i);
+ if (AbstractSet<Element, Container>::
+ getQuantizer().getQuantizedAbsoluteTime(e) != m_time) {
+ return false;
+ }
+ if (e->getSubOrdering() != m_subordering) {
+ return false;
+ }
+
+ // We permit note or rest events etc here, because if a chord is a
+ // little staggered (for performance reasons) then it's not at all
+ // unlikely we could get other events (even rests) in the middle
+ // of it. So long as sample() only permits notes, we should be
+ // okay with this.
+ //
+ // (We're really only refusing things like clef and key events
+ // here, though it's slightly quicker [since most things are
+ // notes] and perhaps a bit safer to do it by testing for
+ // inclusion rather than exclusion.)
+
+ std::string type(e->getType());
+ return (type == Note::EventType ||
+ type == Note::EventRestType ||
+ type == Text::EventType ||
+ type == Indication::EventType ||
+ type == PitchBend::EventType ||
+ type == Controller::EventType ||
+ type == KeyPressure::EventType ||
+ type == ChannelPressure::EventType);
+}
+
+template <class Element, class Container, bool singleStaff>
+bool
+GenericChord<Element, Container, singleStaff>::sample(const Iterator &i,
+ bool goingForwards)
+{
+ Event *e1 = getAsEvent(i);
+ if (!e1->isa(Note::EventType)) {
+ if (goingForwards && m_firstReject == AbstractSet<Element, Container>::getContainer().end()) m_firstReject = i;
+ return false;
+ }
+
+ if (singleStaff) {
+
+ // Two notes that would otherwise be in a chord but are
+ // explicitly in different groups, or have stems pointing in
+ // different directions by design, or have substantially
+ // different x displacements, count as separate chords.
+
+ // Per #930473 ("Inserting notes into beamed chords is
+ // broken"), if one note is in a group and the other isn't,
+ // that's no problem. In fact we should actually modify the
+ // one that isn't so as to suggest that it is.
+
+ if (AbstractSet<Element, Container>::m_baseIterator != AbstractSet<Element, Container>::getContainer().end()) {
+
+ Event *e0 = getAsEvent(AbstractSet<Element, Container>::m_baseIterator);
+
+ if (!(m_stemUpProperty == PropertyName::EmptyPropertyName)) {
+
+ if (e0->has(m_stemUpProperty) &&
+ e1->has(m_stemUpProperty) &&
+ isPersistent__Bool(e0, m_stemUpProperty) &&
+ isPersistent__Bool(e1, m_stemUpProperty) &&
+ get__Bool(e0, m_stemUpProperty) !=
+ get__Bool(e1, m_stemUpProperty)) {
+
+ if (goingForwards && m_firstReject == AbstractSet<Element, Container>::getContainer().end())
+ m_firstReject = i;
+ return false;
+ }
+ }
+
+ long dx0 = 0, dx1 = 0;
+ get__Int(e0, BaseProperties::DISPLACED_X, dx0);
+ get__Int(e1, BaseProperties::DISPLACED_X, dx1);
+ if (abs(dx0 - dx1) >= 700) {
+ if (goingForwards && m_firstReject == AbstractSet<Element, Container>::getContainer().end())
+ m_firstReject = i;
+ return false;
+ }
+
+ if (e0->has(BaseProperties::BEAMED_GROUP_ID)) {
+ if (e1->has(BaseProperties::BEAMED_GROUP_ID)) {
+ if (get__Int(e1, BaseProperties::BEAMED_GROUP_ID) !=
+ get__Int(e0, BaseProperties::BEAMED_GROUP_ID)) {
+ if (goingForwards && m_firstReject == AbstractSet<Element, Container>::getContainer().end())
+ m_firstReject = i;
+ return false;
+ }
+ } else {
+ copyGroupProperties(e0, e1); // #930473
+ }
+ } else {
+ if (e1->has(BaseProperties::BEAMED_GROUP_ID)) {
+ copyGroupProperties(e1, e0); // #930473
+ }
+ }
+ }
+ }
+
+ AbstractSet<Element, Container>::sample(i, goingForwards);
+ push_back(i);
+ return true;
+}
+
+template <class Element, class Container, bool singleStaff>
+void
+GenericChord<Element, Container, singleStaff>::copyGroupProperties(Event *e0,
+ Event *e1) const
+{
+ if (e0->has(BaseProperties::BEAMED_GROUP_TYPE)) {
+ setMaybe__String(e1, BaseProperties::BEAMED_GROUP_TYPE,
+ get__String(e0, BaseProperties::BEAMED_GROUP_TYPE));
+ }
+ if (e0->has(BaseProperties::BEAMED_GROUP_ID)) {
+ setMaybe__Int(e1, BaseProperties::BEAMED_GROUP_ID,
+ get__Int(e0, BaseProperties::BEAMED_GROUP_ID));
+ }
+ if (e0->has(BaseProperties::BEAMED_GROUP_TUPLET_BASE)) {
+ setMaybe__Int(e1, BaseProperties::BEAMED_GROUP_TUPLET_BASE,
+ get__Int(e0, BaseProperties::BEAMED_GROUP_TUPLET_BASE));
+ }
+ if (e0->has(BaseProperties::BEAMED_GROUP_TUPLED_COUNT)) {
+ setMaybe__Int(e1, BaseProperties::BEAMED_GROUP_TUPLED_COUNT,
+ get__Int(e0, BaseProperties::BEAMED_GROUP_TUPLED_COUNT));
+ }
+ if (e0->has(BaseProperties::BEAMED_GROUP_UNTUPLED_COUNT)) {
+ setMaybe__Int(e1, BaseProperties::BEAMED_GROUP_UNTUPLED_COUNT,
+ get__Int(e0, BaseProperties::BEAMED_GROUP_UNTUPLED_COUNT));
+ }
+}
+
+
+template <class Element, class Container, bool singleStaff>
+int
+GenericChord<Element, Container, singleStaff>::getMarkCountForChord() const
+{
+ // need to weed out duplicates
+
+ std::set<Mark> cmarks;
+
+ for (unsigned int i = 0; i < std::vector<typename Container::iterator>::size(); ++i) {
+
+ Event *e = getAsEvent((*this)[i]);
+ std::vector<Mark> marks(Marks::getMarks(*e));
+
+ for (std::vector<Mark>::iterator j = marks.begin(); j != marks.end(); ++j) {
+ cmarks.insert(*j);
+ }
+ }
+
+ return cmarks.size();
+}
+
+
+template <class Element, class Container, bool singleStaff>
+std::vector<Mark>
+GenericChord<Element, Container, singleStaff>::getMarksForChord() const
+{
+ std::vector<Mark> cmarks;
+
+ for (unsigned int i = 0; i < std::vector<typename Container::iterator>::size(); ++i) {
+
+ Event *e = getAsEvent((*this)[i]);
+ std::vector<Mark> marks(Marks::getMarks(*e));
+
+
+ for (std::vector<Mark>::iterator j = marks.begin(); j != marks.end(); ++j) {
+
+ // We permit multiple identical fingering marks per chord,
+ // but not any other sort
+ if (Marks::isFingeringMark(*j) ||
+ std::find(cmarks.begin(), cmarks.end(), *j) == cmarks.end()) {
+ cmarks.push_back(*j);
+ }
+ }
+ }
+
+ return cmarks;
+}
+
+
+template <class Element, class Container, bool singleStaff>
+std::vector<int>
+GenericChord<Element, Container, singleStaff>::getPitches() const
+{
+ std::vector<int> pitches;
+
+ for (typename std::vector<typename Container::iterator>::const_iterator
+ i = std::vector<typename Container::iterator>::begin(); i != std::vector<typename Container::iterator>::end(); ++i) {
+ if (getAsEvent(*i)->has(BaseProperties::PITCH)) {
+ int pitch = get__Int
+ (getAsEvent(*i), BaseProperties::PITCH);
+ if (pitches.size() > 0 && pitches[pitches.size()-1] == pitch)
+ continue;
+ pitches.push_back(pitch);
+ }
+ }
+
+ return pitches;
+}
+
+
+template <class Element, class Container, bool singleStaff>
+bool
+GenericChord<Element, Container, singleStaff>::contains(const Iterator &itr) const
+{
+ for (typename std::vector<typename Container::iterator>::const_iterator
+ i = std::vector<typename Container::iterator>::begin();
+ i != std::vector<typename Container::iterator>::end(); ++i) {
+ if (*i == itr) return true;
+ }
+ return false;
+}
+
+
+template <class Element, class Container, bool singleStaff>
+typename GenericChord<Element, Container, singleStaff>::Iterator
+GenericChord<Element, Container, singleStaff>::getPreviousNote()
+{
+ Iterator i(AbstractSet<Element, Container>::getInitialElement());
+ while (1) {
+ if (i == AbstractSet<Element, Container>::getContainer().begin()) return AbstractSet<Element, Container>::getContainer().end();
+ --i;
+ if (getAsEvent(i)->isa(Note::EventType)) {
+ return i;
+ }
+ }
+}
+
+
+template <class Element, class Container, bool singleStaff>
+typename GenericChord<Element, Container, singleStaff>::Iterator
+GenericChord<Element, Container, singleStaff>::getNextNote()
+{
+ Iterator i(AbstractSet<Element, Container>::getFinalElement());
+ while ( i != AbstractSet<Element, Container>::getContainer().end() &&
+ ++i != AbstractSet<Element, Container>::getContainer().end()) {
+ if (getAsEvent(i)->isa(Note::EventType)) {
+ return i;
+ }
+ }
+ return AbstractSet<Element, Container>::getContainer().end();
+}
+
+
+template <class Element, class Container, bool singleStaff>
+typename GenericChord<Element, Container, singleStaff>::Iterator
+GenericChord<Element, Container, singleStaff>::getFirstElementNotInChord()
+{
+ return m_firstReject;
+}
+
+
+template <class Element, class Container, bool singleStaff>
+bool
+GenericChord<Element, Container, singleStaff>::PitchGreater::operator()(const Iterator &a,
+ const Iterator &b)
+{
+ try {
+ long ap = get__Int(getAsEvent(a), BaseProperties::PITCH);
+ long bp = get__Int(getAsEvent(b), BaseProperties::PITCH);
+ return (ap < bp);
+ } catch (Event::NoData) {
+ std::cerr << "Bad karma: PitchGreater failed to find one or both pitches" << std::endl;
+ return false;
+ }
+}
+
+
+typedef GenericChord<Event, Segment, true> Chord;
+typedef GenericChord<Event, CompositionTimeSliceAdapter, false> GlobalChord;
+
+
+}
+
+
+#endif
+