
    rha                      S r SSKJr  SSKJr  SSKrSSKrSSKrSSKrSSK	r	SSK
rSSKrSSKJrJrJr  SSKrSSKJr  SSKJr  SSKJr  SS	KJr  SS
KJr  SSKJr  SSKJr  SSKJr  SSKJr  SSKJr  SSKJr  SSKJ r   SSKJ!r!  SSKJ"r"  SSKJ#r#  SSKJ$r$  SSKJ%r%  SSKJ&r&  SSKJ'r'  SSKJ(r(  SSKJ)r)  SSKJ*r*  SSKJ+r+  SSKJ,r,  SSKJ-r-  SSKJ.r.  SS K/J0r0  SS!KJ1r1  SS"KJ2r2  SS#KJ3r3  SS$KJ4r4  SS%K5J6r6  SS&K7J8r8  SS'K5J9r9  SS(K:J;r;  SS)K:J<r<  \ Rz                  " S*5      r>\6R~                  r@\6R                  rB\R                  (       a  SS+KDJErE  SS,KJFrF  SS-KJGrG  S?S. jrHS@S/ jrISAS0 jrJ SBSS1S2.             SCS3 jjjrK " S4 S55      rL " S6 S75      rM " S8 S9\M\85      rN " S: S;\M5      rO " S< S=\M5      rP\QS>:X  a  SSKr\R                  " 5         gg)Dz>
Converters for music21 objects to musicxml using ElementTree
    )annotations)OrderedDictN)Element
SubElementComment)articulations)base)bar)clef)chord)common)AppendSpanners)opFrac)defaultsduration)dynamics)environment)exceptions21)expressions)harmony)
instrument)key)layout)metadata)note)meter)pitch)prebase)repeat)spanner)stream)OffsetIterator)style)tempo)text)tie)helpers)PartStaffExporterMixin)
xmlObjectsMusicXMLExportException)MusicXMLWarningzmusicxml.m21ToXml)OffsetQL)roman)	tablaturec                    U S:X  a  gU S:X  a  [        S5      eU S:X  a  [        S5      eU S:X  a  [        S5      eU S	:X  a  [        S
5      eU S:X  a  [        S5      eU $ )a  
Convert a music21 type to a MusicXML type or raise a MusicXMLExportException

>>> musicxml.m21ToXml.typeToMusicXMLType('longa')
'long'
>>> musicxml.m21ToXml.typeToMusicXMLType('quarter')
'quarter'
>>> musicxml.m21ToXml.typeToMusicXMLType('duplex-maxima')
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
Cannot convert "duplex-maxima" duration to MusicXML (too long).
>>> musicxml.m21ToXml.typeToMusicXMLType('inexpressible')
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
Cannot convert inexpressible durations to MusicXML.
>>> musicxml.m21ToXml.typeToMusicXMLType('zero')
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
Cannot convert durations without types to MusicXML.
longalong2048thz9Cannot convert "2048th" duration to MusicXML (too short).zduplex-maximaz?Cannot convert "duplex-maxima" duration to MusicXML (too long).inexpressiblez3Cannot convert inexpressible durations to MusicXML.complexzyCannot convert complex durations to MusicXML. Try exporting with makeNotation=True or manually running splitAtDurations()zeroz3Cannot convert durations without types to MusicXML.r+   )values    S/home/james-whalen/.local/lib/python3.13/site-packages/music21/musicxml/m21ToXml.pytypeToMusicXMLTyper:   U   s    , 	(	%&abb	/	!%MO 	O	/	!%&[\\	)	%\] 	] 
&%&[\\    c                    U (       d  gSU ;  a$  [         R                  " U 5      R                  5       $ U R                  5       $ )z
Normalize a css3 name to hex or leave it alone.

>>> musicxml.m21ToXml.normalizeColor('')
''
>>> musicxml.m21ToXml.normalizeColor('red')
'#FF0000'
>>> musicxml.m21ToXml.normalizeColor('#00ff00')
'#00FF00'
 #)	webcolorsname_to_hexupper)colors    r9   normalizeColorrC   ~   s8     
%$$U+1133{{}r;   c                    U R                   nUb  U$ U R                  5        H2  nUR                  R                   c  M  UR                  R                   s  $    g)a  
Get metadata from site or context, so that a Part
can be shown and have the rich metadata of its Score

>>> s = stream.Score()
>>> s2 = s.transpose(4)
>>> md = metadata.Metadata()
>>> md.title = 'emptiness'
>>> s.metadata = md
>>> s2.metadata is None
True
>>> musicxml.m21ToXml.getMetadataFromContext(s2).title
'emptiness'
>>> musicxml.m21ToXml.getMetadataFromContext(s).title
'emptiness'
>>> p = stream.Part()
>>> s2.insert(0, p)
>>> musicxml.m21ToXml.getMetadataFromContext(p).title
'emptiness'
N)r   contextSitessite)smdcontextSites      r9   getMetadataFromContextrJ      sQ    . 
B	~	~~'$$0##,,, ( r;   F)	transform
forceEmptyc                   Uc  [         R                  " U5      n [        X5      nUb  U" U5      nUS;   a  USLa  g[	        X5      nUb  [        U5      Ul        U$ ! [         a     gf = f)a$  
If m21El has an attribute called attributeName, create a new SubElement
for xmlEl and set its text to the value of the m21El attribute.

Pass a function or lambda function as transform to transform the
value before setting it.  String transformation is assumed.

Returns the SubElement

Will not create an empty element unless forceEmpty is True

>>> from xml.etree.ElementTree import Element
>>> e = Element('accidental')

>>> seta = musicxml.m21ToXml._setTagTextFromAttribute
>>> acc = pitch.Accidental()
>>> acc.alter = -2.0

>>> subEl = seta(acc, e, 'alter')
>>> subEl.text
'-2.0'
>>> subEl in e
True

>>> XB = musicxml.m21ToXml.XMLExporterBase()
>>> XB.dump(e)
<accidental>
  <alter>-2.0</alter>
</accidental>

add a transform

>>> subEl = seta(acc, e, 'alter', transform=int)
>>> subEl.text
'-2'

NNr=   T)r   hyphenToCamelCasegetattrAttributeErrorr   strr&   )m21ElxmlEltagattributeNamerK   rL   r8   
subElements           r9   _setTagTextFromAttributerX      s    \ 005- % 
z5E'Je*
  s   A 
A*)A*c                      \ rS rSrSr\" / SQ5      rSSS jjrSSS jjrSS jr	SS jr
S	 rS
 rS rS rS rS rS rS rS rS rS rSS jrSrg)GeneralObjectExporter   zh
Packs any `Music21Object` into a well-formed score and exports
a bytes object MusicXML representation.
))Score	fromScore)PartfromPart)MeasurefromMeasure)Voice	fromVoice)Stream
fromStream)GeneralNotefromGeneralNote)Pitch	fromPitch)DurationfromDuration)DynamicfromDynamic)DiatonicScalefromDiatonicScale)Scale	fromScale)Music21ObjectfromMusic21ObjectNc                    Xl         SU l        g NT
generalObjmakeNotation)selfobjs     r9   __init__GeneralObjectExporter.__init__  s    "&r;   c                   Uc  U R                   nUc  [        S5      eU R                  (       a"  U R                  U5      nU R	                  U5      $ [        U[        R                  5      (       d  [        S5      eU R	                  U5      $ )a  
Return a bytes object representation of anything from a
Score to a single pitch.

When :attr:`makeNotation` is True (default), wraps `obj` in a well-formed
`Score`, makes a copy, and runs :meth:`~music21.stream.base.Stream.makeNotation`
on each of the parts. To skip copying and making notation, set `.makeNotation`
on this instance to False.

>>> p = pitch.Pitch('D#4')
>>> GEX = musicxml.m21ToXml.GeneralObjectExporter(p)
>>> out = GEX.parse()  # out is bytes
>>> outStr = out.decode('utf-8')  # now is string
>>> print(outStr.strip())
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE score-partwise
  PUBLIC "-//Recordare//DTD MusicXML ... Partwise//EN"
  "http://www.musicxml.org/dtds/partwise.dtd">
<score-partwise version="...">
  <movement-title>Music21 Fragment</movement-title>
  <identification>
    <creator type="composer">Music21</creator>
    <encoding>
      <encoding-date>...</encoding-date>
      <software>music21 v...</software>
    </encoding>
  </identification>
  <defaults>
    <scaling>
      <millimeters>7</millimeters>
      <tenths>40</tenths>
    </scaling>
  </defaults>
  <part-list>
    <score-part id="...">
      <part-name />
    </score-part>
  </part-list>
  <!--=========================== Part 1 ===========================-->
  <part id="...">
    <!--========================= Measure 1 ==========================-->
    <measure implicit="no" number="1">
      <attributes>
        <divisions>10080</divisions>
        <time>
          <beats>1</beats>
          <beat-type>4</beat-type>
        </time>
        <clef>
          <sign>G</sign>
          <line>2</line>
        </clef>
      </attributes>
      <note>
        <pitch>
          <step>D</step>
          <alter>1</alter>
          <octave>4</octave>
        </pitch>
        <duration>10080</duration>
        <type>quarter</type>
        <accidental>sharp</accidental>
      </note>
    </measure>
  </part>
</score-partwise>
zMust have an object to exportz.Can only export Scores with makeNotation=False)rw   r,   rx   fromGeneralObjectparseWellformedObject
isinstancer"   r\   )ry   rz   outObjs      r9   parseGeneralObjectExporter.parse  s|    H ;//C;)*IJJ++C0F--f55c6<<00-.^__--c22r;   c                h    [        XR                  S9nUR                  5         UR                  5       $ )z
Parse an object that has already gone through the
`.fromGeneralObject` conversion, which has produced a copy with
:meth:`~music21.stream.Score.makeNotation` run on it
(unless :attr:`makeNotation` is False).

Returns bytes.
)rx   )ScoreExporterrx   r   asBytes)ry   scscoreExporters      r9   r   +GeneralObjectExporter.parseWellformedObject`  s/     &b7H7HI$$&&r;   c                v   UR                   nSn[        U[        R                  5      (       a0  U R                  (       a  UR                  SUR                  /SSSSS9nU R                  R                  5        H  u  pEXB;   d  M  [        X5      nU" U5      n  O   Uc  [        SU R                   S3-   5      eU$ )a?  
Converts any Music21Object (or a duration or a pitch) to something that
can be passed to :class:`ScoreExporter`.

>>> GEX = musicxml.m21ToXml.GeneralObjectExporter()
>>> s = GEX.fromGeneralObject(duration.Duration(3.0))
>>> s
<music21.stream.Score 0x...>
>>> s.show('t')
{0.0} <music21.stream.Part 0x...>
    {0.0} <music21.stream.Measure 1 offset=0.0>
        {0.0} <music21.clef.TrebleClef>
        {0.0} <music21.meter.TimeSignature 6/8>
        {0.0} <music21.note.Note C>
        {3.0} <music21.bar.Barline type=final>
>>> s[note.NotRest].first().duration
<music21.duration.Duration 3.0>

* Changed in v8: fills gaps with rests before calling makeNotation
  to avoid duplicating effort with :meth:`PartExporter.fixupNotationMeasured`.

>>> v = stream.Voice(note.Note())
>>> m = stream.Measure([meter.TimeSignature(), v])
>>> GEX = musicxml.m21ToXml.GeneralObjectExporter(m)
>>> out = GEX.parse()  # out is bytes
>>> outStr = out.decode('utf-8')  # now is string
>>> '<note print-object="no" print-spacing="yes">' in outStr
True
>>> len(v[note.Rest])  # original stream unchanged
0
N        TF)refStreamOrTimeRangefillGapsinPlace	hideReststimeRangeFromBarDurationzCannot translate the object z; to a complete musicXML document; put it in a Stream first!)classesr   r"   rd   rx   	makeRestshighestTimeclassMappingitemsrP   r,   rw   )ry   rz   r   r   cMmethNamemeths          r9   r~   'GeneralObjectExporter.fromGeneralObjectm  s    @ ++c6==))d.?.?--c3??5K)-(-*.9=	   "C !--335LB}t.c	 6
 >).__%%`ab  r;   c                    U R                   (       a  UR                  SS9  UR                  5       (       d  [        R                  " U S3[        S9  U$ )z>
Runs :meth:`~music21.stream.Score.makeNotation` on the copy.
Tr   z/ is not well-formed; see isWellFormedNotation())category)rx   isWellFormedNotationwarningswarnr-   )ry   r   s     r9   r]   GeneralObjectExporter.fromScore  sJ     OODO)&&((MMRD OP(* 	r;   c                    UR                   (       a  UR                  SS9  [        R                  " 5       nUR	                  SU5        [
        R                  " [        U5      5      Ul        U R                  U5      $ )z%
From a part, put it in a new score.
Tr   r   )
isFlatmakeMeasuresr"   r\   insertcopydeepcopyrJ   r   r]   )ry   prG   s      r9   r_   GeneralObjectExporter.fromPart  sV     88NN4N(LLN	A]]#9!#<=
~~a  r;   c                   UR                  5         UR                  SS9  UR                  R                  c  SUR                  l        UR	                  [
        R                  [        R                  R                  R                  S9nUc  [
        R                  " USS9Ul        OX!l        [        R                  " 5       nUR                  U5        [        R                   " [#        U5      5      Ul        UR	                  [        R                  5      nUb"  UR&                  Ul        UR(                  Ul        U R+                  U5      $ )z
Translate a music21 Measure into a
complete MusicXML string representation.

Note: this method is called for complete MusicXML
representation of a Measure, not for partial
solutions in Part or Stream production.
Tr   measure)getElementMethodrecurse)coreGatherMissingSpannersrx   r$   measureNumberinggetContextByClassr   Clefr   enumsElementSearchAT_OR_BEFORE_OFFSETbestClefr"   r^   appendr   r   rJ   r   partNamepartAbbreviationr_   )ry   m"clef_from_measure_start_or_contextr   context_parts        r9   ra   !GeneralObjectExporter.fromMeasure  s     	
##%	t$77##+'0AGG$-.-@-@II#\\77KK .A .
* .5]]1d3AF7FKKM	]]#9!#<=
**6;;7#%..AJ!-!>!>A}}Qr;   c                p    [         R                  " SS9nUR                  SU5        U R                  U5      $ )N   numberr   )r"   r`   r   ra   )ry   vr   s      r9   rc   GeneralObjectExporter.fromVoice  s.    NN!$	A""r;   c                   UR                   (       a  [        R                  " 5       nUR                  U5        [        R
                  " U5      Ul        UR                  [        R                  5      R                  S5      (       d  [        R                  " U5      Ul        UR                  SS9  [        R
                  " [        U5      5      Ul        U R                  U5      $ UR!                  5       (       a  [        R"                  " 5       nUR                  U5        [        R
                  " U5      Ul        UR                  SS9  [        R
                  " [        U5      5      Ul        U R%                  U5      $ UR                  [        R&                  5      R)                  5       R                   (       a  [        R                  " 5       nUR                  U5        [        R
                  " U5      Ul        UR                  [        R                  5      R                  S5      (       d  SnOSnUR                  SUS9  [        R
                  " [        U5      5      Ul        U R                  U5      $ UR                  [        R                  5      R                  S5      (       d  SnOSnUR                  SUS9nU R%                  U5      $ )Nr   Tr   F)r   r   )r   r"   r^   mergeAttributesr   r   elementsgetElementsByClassr   r   getElementsByOffsetr   rx   rJ   r   r_   hasPartLikeStreamsr\   r]   rd   first)ry   stst2r   s       r9   re    GeneralObjectExporter.fromStream  s   99++-C#==,CL((3GGLL==-T*==)?)CDCL==%%""$$,,.C#==,CLT*==)?)CDCL>>#&&""6==1779@@++-C#==,CL((3GGLL TH===)?)CDCL==%% ((3GGLL //%(/CC>>"%%r;   c                    [         R                  " U5      n[        R                  " 5       nX#l        U R                  U5      $ )z
Translate a music21 :class:`~music21.duration.Duration` into
a complete MusicXML representation.

Rarely, rarely used.  Only if you call .show() on a duration object
)r   r   r   Noter   rg   )ry   ddCopyns       r9   rk   "GeneralObjectExporter.fromDuration	  s3     a IIK
##A&&r;   c                    [         R                  " U5      n[        R                  " 5       nUR	                  U5        U R                  U5      $ )z^
Provide a complete MusicXML string from a single dynamic by
putting it into a Stream first.

)r   r   r"   rd   r   re   )ry   dynamicObjectr   outs       r9   rm   !GeneralObjectExporter.fromDynamic  s8     m,mmo

5s##r;   c                   [         R                  " SS9n[        SUR                  R	                  5       S-   5       H  nUR                  U5      n[        R                  " 5       nXEl        US:X  a  UR                  UR                  5        UR                  UR                  5       R                  :X  a  SUl        OSUl        UR                  U5        M     UR                  5       Ul        U R!                  U5      $ )a  
Generate the pitches from this scale
and put it into a stream.Measure, then call
fromMeasure on it.

>>> cMaj = scale.MajorScale('C')
>>> GEX = musicxml.m21ToXml.GeneralObjectExporter()
>>> m = GEX.fromScale(cMaj)
>>> m
<music21.stream.Score 0x11d4f17b8>

>>> m.show('text')
{0.0} <music21.stream.Part 0x116a04b38>
    {0.0} <music21.stream.Measure 1 offset=0.0>
        {0.0} <music21.clef.TrebleClef>
        {0.0} <music21.meter.TimeSignature 10/4>
        {0.0} <music21.note.Note C>
        {4.0} <music21.note.Note D>
        {5.0} <music21.note.Note E>
        {6.0} <music21.note.Note F>
        {7.0} <music21.note.Note G>
        {8.0} <music21.note.Note A>
        {9.0} <music21.note.Note B>
r   r      )r"   r`   rangeabstractgetDegreeMaxUniquepitchFromDegreer   r   r   addLyricnamegetTonicquarterLengthr   bestTimeSignaturetimeSignaturera   )ry   scaleObjectr   ir   r   s         r9   rq   GeneralObjectExporter.fromScale%  s    4 NN!$q+..AACaGHA++A.A		AGAv

;++,vv--/444"#"#HHQK I --/""r;   c                \   [         R                  " SS9n[        SUR                  R	                  5       S-   5       H  nUR                  U5      n[        R                  " 5       nXEl        US:X  a  UR                  UR                  5        UR                  UR                  5       R                  :X  a  SUl        O7UR                  UR                  5       R                  :X  a  SUl        OSUl        UR                  U5        M     UR                  5       Ul        U R#                  U5      $ )a  
Return a complete musicxml of the DiatonicScale

Overrides the general scale behavior to highlight
the tonic and dominant.

>>> cMaj = scale.MajorScale('C')
>>> GEX = musicxml.m21ToXml.GeneralObjectExporter()
>>> m = GEX.fromDiatonicScale(cMaj)
>>> m
<music21.stream.Score 0x11d4f17b8>

>>> m.show('text')
{0.0} <music21.stream.Part 0x116a04b38>
    {0.0} <music21.stream.Measure 1 offset=0.0>
        {0.0} <music21.clef.TrebleClef>
        {0.0} <music21.meter.TimeSignature 11/4>
        {0.0} <music21.note.Note C>
        {4.0} <music21.note.Note D>
        {5.0} <music21.note.Note E>
        {6.0} <music21.note.Note F>
        {7.0} <music21.note.Note G>
        {9.0} <music21.note.Note A>
        {10.0} <music21.note.Note B>
r   r   r      )r"   r`   r   r   r   r   r   r   r   r   r   r   r   getDominantr   r   r   ra   )ry   diatonicScaleObjectr   r   r   r   s         r9   ro   'GeneralObjectExporter.fromDiatonicScaleO  s    6 NN!$q-66IIKaOPA#33A6A		AGAv

.334vv,557<<<"#.::<AAA"#"#HHQK Q --/""r;   c                    [         R                  " U5      n[        R                  " SS9nUR	                  U5        U R                  U5      $ )z6
return a single TimeSignature as a musicxml document
r   r   )r   r   r"   r`   r   ra   )ry   rz   objCopyr   s       r9   rs   'GeneralObjectExporter.fromMusic21Object|  s?    
 --$ nnA&

7$$r;   c                   [         R                  " U5      n[        R                  " U5      nSUR                  s=:  a  S::  a)  O  O&UR                  S[        R                  " U5      5        [        R                  R                  USSUR                  /S9  [        R                  R                  USS9  U R                  U5      $ )ay  
Translate a music21 :class:`~music21.note.Note` into an object
ready to be parsed.

An attempt is made to find the best TimeSignature for quarterLengths
<= 6:

>>> n = note.Note('c3')
>>> n.quarterLength = 3
>>> GEX = musicxml.m21ToXml.GeneralObjectExporter()
>>> sc = GEX.fromGeneralNote(n)
>>> sc.show('t')
{0.0} <music21.stream.Part 0x1046afa90>
    {0.0} <music21.stream.Measure 1 offset=0.0>
        {0.0} <music21.clef.BassClef>
        {0.0} <music21.meter.TimeSignature 6/8>
        {0.0} <music21.note.Note C>
        {3.0} <music21.bar.Barline type=final>

But longer notes will be broken into tied components placed in
4/4 measures:

>>> long_note = note.Note('e5', quarterLength=40)
>>> GEX = musicxml.m21ToXml.GeneralObjectExporter()
>>> sc = GEX.fromGeneralNote(long_note)
>>> sc[meter.TimeSignature].first()
<music21.meter.TimeSignature 4/4>
>>> len(sc[stream.Measure])
10
r   g      @T)r   r   r   )r   r   r"   r^   r   r   r   r   rx   r   makeTupletBracketsr_   )ry   r   nCopynew_parts       r9   rg   %GeneralObjectExporter.fromGeneralNote  s    D a ;;u%q%#%OOAu66x@A(("#U%8%8!9 	) 	

 	..x.F}}X&&r;   c                    [         R                  " 5       n[        R                  " U5      Ul        [
        R                  " SS9nUR                  U5        U R                  U5      $ )a  
Translate a music21 :class:`~music21.pitch.Pitch` into an object
ready to be parsed.

>>> p = pitch.Pitch('c#3')
>>> GEX = musicxml.m21ToXml.GeneralObjectExporter()
>>> sc = GEX.fromPitch(p)
>>> sc.show('t')
{0.0} <music21.stream.Part 0x1046afa90>
    {0.0} <music21.stream.Measure 1 offset=0.0>
        {0.0} <music21.clef.BassClef>
        {0.0} <music21.meter.TimeSignature 1/4>
        {0.0} <music21.note.Note C#>
r   r   )	r   r   r   r   r   r"   r`   r   ra   )ry   r   r   r   s       r9   ri   GeneralObjectExporter.fromPitch  sI      IIK--"nnA&

1$$r;   rv   N)rz   prebase.ProtoM21Object | None)rz   r   returnbytes)r   zstream.Scorer   r   )rz   zprebase.ProtoM21Objectr   zpitch.Pitch)__name__
__module____qualname____firstlineno____doc__r   r   r{   r   r   r~   r]   r_   ra   rc   re   rk   rm   rq   ro   rs   rg   ri   __static_attributes__ r;   r9   rZ   rZ      so        L 'O3b'6p

! >#
&&P' 
$(#T+#Z%,'\%r;   rZ   c                      \ rS rSrSrSS jrSSS jjrSS S jjr\S 5       r	SS jr
S!S	 jrS
 rS rS rS rS rS rS rS rS!S jrS!S jrS!S jrS!S jrS!S jrSS.S"S jjrS#S jrSrg)$XMLExporterBasei  za
contains functions that could be called
at multiple levels of exporting (Score, Part, Measure).
c                X    [        S5      U l        [        R                  " 5       U l        g )Nzoverride-me-in-subclasses)r   xmlRootr"   rd   ry   s    r9   r{   XMLExporterBase.__init__  s    :;%+]]_r;   c                .   [         R                  " 5       nUR                  U R                  5       5        U R                  n[
        R                  " X1S9nUR                  UR                  S5      5        UR                  5       nUR                  5         U$ )z
returns the xmlRoot as a bytes object. If noCopy is True
(default), modifies the file for pretty-printing in place.  Otherwise,
make a copy.
)noCopyutf-8)
ioBytesIOwrite	xmlHeaderr  r(   
dumpStringencodegetvalueclose)ry   r  siorootObjrootObj_stringr   s         r9   r   XMLExporterBase.asBytes  sk     jjl		$.."#,, ++GC		.''01LLN		r;   c                   [        [        U5      S5      n[        R                  " SU-
  S-  5      n[        R                  " SU-
  S-  5      nSU-  S-   U-   S-   SU-  -   n[        U5      nU R                  R                  U5        g)a  
Add a divider to xmlRoot.

>>> from xml.etree.ElementTree import Element
>>> e1 = Element('accidental')
>>> e2 = Element('accidental')

>>> XB = musicxml.m21ToXml.ScoreExporter()
>>> XB.xmlRoot.append(e1)
>>> XB.addDividerComment('second accidental below')
>>> XB.xmlRoot.append(e2)
>>> XB.dump(XB.xmlRoot)
<score-partwise version="...">
  <accidental />
  <!--================== second accidental below ===================-->
  <accidental />
  </score-partwise>
<   r   = N)minlenmathfloorceilr   r  r   )ry   commentcommentLengthspacerLengthLowspacerLengthHighcommentTextdividers          r9   addDividerComment!XMLExporterBase.addDividerComment  s    & CL"-**b=&8A%=>99b=&8A%=>_,3g=CsM]G]^+&G$r;   c                .    [         R                  " U 5      $ r   )r(   dump)rz   s    r9   r(  XMLExporterBase.dump  s    ||C  r;   c                R    S[         R                  R                  S5      -   S-   S-   $ )Nsd   <?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE score-partwise  PUBLIC "-//Recordare//DTD MusicXML r	  s    Partwise//EN" s-   "http://www.musicxml.org/dtds/partwise.dtd">
)r   musicxmlVersionr  r  s    r9   r  XMLExporterBase.xmlHeader  s<    =**11':; )) H	H 	Ir;   Nc                .   [        U[        R                  5      (       a  UnOUR                  SL a  gUR                  n[        R
                  " U5      (       d  U4nUc
  S U 5       nO[        R
                  " U5      (       d  U4n[        X45       H]  u  pg [        XW5      nU[        R                  ;   a  [        R                  " U5      nUc  M@   [        U5      nUR                  Xh5        M_     g! [         a     Mo  f = f! [         a     M  f = f)zm
Sets any attribute from .style, doing some conversions.

m21Object can also be a style.Style object itself.
FNc              3  N   #    U  H  n[         R                  " U5      v   M     g 7fr   )r   rO   ).0xs     r9   	<genexpr>5XMLExporterBase.setStyleAttributes.<locals>.<genexpr>"  s     K]0033]s   #%)r   r$   StylehasStyleInformationr   
isIterableziprP   rQ   r*   STYLE_ATTRIBUTES_YES_NO_TO_BOOLbooleanToYesNorR   
ValueErrorset)	ry   mxObject	m21ObjectmusicXMLNamesm21NamesstObjxmlNamem21Namem21Values	            r9   setStyleAttributes"XMLExporterBase.setStyleAttributes  s    i--E**e3OOE  //*,MK]KH""8,, {H #M <G"52 *DDD%44X>x= LL+# != "   s$   C5D5
DD
DDc                R    SnSnU R                  XX45        U R                  X5        g)a<  
sets the justification, print-style-align group, and
text-decoration, text-rotation,
letter-spacing, line-height, lang, text-direction, and
enclosure, on an
m21Object, which must have style.TextStyle as its Style class,
and then calls setPrintStyleAlign

conforms to attr-group %text-formatting in the MusicXML DTD
)justifyztext-decorationztext-rotationzletter-spacingzline-heightlangztext-direction	enclosure)rF  textDecorationtextRotationletterSpacing
lineHeightlanguagetextDirectionrH  N)rC  setPrintStyleAlignry   r;  r<  r=  r>  s        r9   setTextFormatting!XMLExporterBase.setTextFormatting9  s0    OL]M4r;   c                N    U R                  X5        U R                  UUSS5        g)z
runs setPrintStyle and then sets horizontalAlign and verticalAlign, on an
m21Object, which must have style.TextStyle as its Style class.

conforms to attr-group %print-style-align in the MusicXML DTD
)valignhalign)alignVerticalalignHorizontalN)setPrintStylerC  ry   r;  r<  s      r9   rO  "XMLExporterBase.setPrintStyleAlignK  s,     	8/ ) 4 D	Fr;   c                j    U R                  X5        U R                  X5        U R                  X5        g)z
get position, font, and color information from the mxObject
into the m21Object, which must have style.TextStyle as its Style class.

conforms to attr-group %print-style in the MusicXML DTD
N)setPositionsetFontsetColorrY  s      r9   rX  XMLExporterBase.setPrintStyleX  s*     	-X)h*r;   c                    [        U[        R                  5      (       a  UnOUR                  (       a  UR                  nOgUR                  (       a  UR                  SS5        gg)z
sets print-object to 'no' if m21Object.style.hideObjectOnPrint is True
or if m21Object is a StyleObject and has .hideObjectOnPrint set to True.
Nprint-objectno)r   r$   r3  r4  hideObjectOnPrintr:  )ry   r;  r<  r   s       r9   setPrintObjectXMLExporterBase.setPrintObjectc  sK    
 i--B**BLL.  r;   c                    U R                  XSS5        SUR                  ;   a&  [        UR                  S   5      UR                  S'   gg)zI
Sets mxObject['color'] to a normalized version of m21Object.style.color
rB   N)rC  attribrC   rY  s      r9   r^  XMLExporterBase.setColorr  sC     	WgFhoo%'5hoog6N'OHOOG$ &r;   c                   SnSnU R                  XX45        [        U[        R                  5      (       a  UnOUR                  (       a  UR                  nOg[        US5      (       as  UR                  S:X  a/  UR                  SS5        UR                  R                  SS5        O4UR                  S:X  a$  UR                  SS5        UR                  SS	5        [        US
5      (       a  UR                  (       an  [        R                  " UR                  5      (       a,  UR                  SSR                  UR                  5      5        gUR                  SUR                  5        ggg)a  
sets font-family, font-style, font-size, and font-weight as
fontFamily (list), fontStyle, fontSize and fontWeight from
an object into a TextStyle object

conforms to attr-group %font in the MusicXML DTD

>>> from xml.etree.ElementTree import fromstring as El
>>> XB = musicxml.m21ToXml.XMLExporterBase()
>>> mxObj = El('<text>hi</text>')
>>> te = expressions.TextExpression('hi!')
>>> te.style.fontFamily = ['Courier', 'monospaced']
>>> te.style.fontStyle = 'italic'
>>> te.style.fontSize = 24.0
>>> XB.setFont(mxObj, te)
>>> XB.dump(mxObj)
<text font-family="Courier,monospaced" font-size="24" font-style="italic">hi</text>

>>> XB = musicxml.m21ToXml.XMLExporterBase()
>>> mxObj = El('<text>hi</text>')
>>> te = expressions.TextExpression('hi!')
>>> te.style.fontStyle = 'bold'
>>> XB.setFont(mxObj, te)
>>> XB.dump(mxObj)
<text font-weight="bold">hi</text>

>>> te.style.fontStyle = 'bolditalic'
>>> XB.setFont(mxObj, te)
>>> XB.dump(mxObj)
<text font-style="italic" font-weight="bold">hi</text>
)
font-stylez	font-sizefont-weight)	fontStylefontSize
fontWeightNrl  boldrk  rj  
bolditalicitalic
fontFamilyzfont-family,)rC  r   r$   r3  r4  hasattrrl  r:  rg  poprr  r   r5  join)ry   r;  r<  r=  r>  r   s         r9   r]  XMLExporterBase.setFont}  s   @ C:]Mi--B**B2{##||v%]F3##L$7-]F3\842|$$  //]CHHR]],CD]BMM:	 *7$r;   c                0    SnSnU R                  XX45        g)z
set positioning information for an mxObject from
default-x, default-y, relative-x, relative-y from
the .style attribute's absoluteX, relativeX, etc. attributes.
)z	default-xz	default-yz
relative-xz
relative-y)	absoluteX	absoluteY	relativeX	relativeYNrC  rP  s        r9   r\  XMLExporterBase.setPosition  s     OG]Mr;   c                0   UR                   SL a  gUR                  nSU;  a  SU;  a  gSnUR                  (       a  UR                  S   nSnO"UR                  (       a  UR                  S   nOgU(       a;  [	        US5      nU R                  Xe5        UR                  b  UR                  Ul        UR                  bW  [	        US5      n[        UR                  5      Ul        UR                  S	[        R                  " UR                  5      5        gg)
a  
>>> from xml.etree.ElementTree import fromstring as El
>>> XB = musicxml.m21ToXml.XMLExporterBase()
>>> mxObj = El('<note />')
>>> n = note.Note('C-5')

Most common case: does nothing

>>> XB.setEditorial(mxObj, n)
>>> XB.dump(mxObj)
<note />

>>> fn = editorial.Comment('flat is obvious error for sharp')
>>> fn.levelInformation = 2
>>> fn.isFootnote = True
>>> n.editorial.footnotes.append(fn)
>>> XB.setEditorial(mxObj, n)
>>> XB.dump(mxObj)
<note>
  <footnote>flat is obvious error for sharp</footnote>
  <level reference="no">2</level>
</note>

Placing information in `.editorial.comments` only puts out the level:

>>> mxObj = El('<note />')
>>> n = note.Note('C-5')
>>> com = editorial.Comment('flat is obvious error for sharp')
>>> com.levelInformation = 'hello'
>>> com.isReference = True
>>> n.editorial.comments.append(com)
>>> XB.setEditorial(mxObj, n)
>>> XB.dump(mxObj)
<note>
  <level reference="yes">hello</level>
</note>
FN	footnotescommentsr   Tfootnotelevel	reference)hasEditorialInformation	editorialr  r  r   rQ  r&   levelInformationrR   r:  r*   r8  isReference)ry   r;  r<  emakeFootnotecmxFnmxLevels           r9   setEditorialXMLExporterBase.setEditorial  s    L ,,5 aJa$7;;AALZZ

1Ah
3D""4+vv!FF	) 73Gq112GLKKZ%>%>q}}%MN *r;   c                    Uc  [        S5      nOUn[        nU" XSS[        R                  S9  U" XS5        U R	                  U5      n[        U5      (       a  UR                  U5        Uc  U$ g)a  
Given a PageLayout object, set object data for <print>

>>> pl = layout.PageLayout()
>>> pl.pageHeight = 4000
>>> pl.isNew = True
>>> pl.rightMargin = 30.25
>>> pl.leftMargin = 20
>>> pl.pageNumber = 5

>>> XPBase = musicxml.m21ToXml.XMLExporterBase()
>>> mxPrint = XPBase.pageLayoutToXmlPrint(pl)
>>> XPBase.dump(mxPrint)
<print new-page="yes" page-number="5">
  <page-layout>
    <page-height>4000</page-height>
    <page-margins>
      <left-margin>20</left-margin>
      <right-margin>30.25</right-margin>
    </page-margins>
  </page-layout>
</print>


>>> MP = musicxml.xmlToM21.MeasureParser()
>>> pl2 = MP.xmlPrintToPageLayout(mxPrint)
>>> pl2.isNew
True
>>> pl2.rightMargin
30.25
>>> pl2.leftMargin
20
>>> pl2.pageNumber
5
>>> pl2.pageHeight
4000
Nprintnew-pageisNewrK   zpage-number)r   setAttributeFromAttributer*   r8  pageLayoutToXmlPageLayoutr  r   )ry   
pageLayout	mxPrintInmxPrintsetbmxPageLayouts         r9   pageLayoutToXmlPrint$XMLExporterBase.pageLayoutToXmlPrint  st    N g&GG(Z*gAZAZ[Z-055jA|NN<(N r;   c                    Uc  [        S5      nOUn[        nU" XS5        U" XS5        [        S5      nS H  nU" XUS-   5        M     [        U5      (       a  UR                  U5        Uc  U$ g)z
get a <page-layout> element from a PageLayout

Called out from pageLayoutToXmlPrint because it
is also used in the <defaults> tag
Nzpage-layoutzpage-heightz
page-widthzpage-margins)leftrighttopbottom-marginr   rX   r  r   )ry   r  mxPageLayoutInr  setamxPageMargins	directions          r9   r  )XMLExporterBase.pageLayoutToXmlPageLayout=  s     !"=1L)L'Z}5Z|4  /;II	,AB <}.! "r;   c                    Uc  [        S5      nOUn[        nU" XSS[        R                  S9  [        S5      nU R	                  X5        [        U5      (       a  UR                  U5        Uc  U$ g)a  
Given a SystemLayout tag, set a <print> tag

>>> sl = layout.SystemLayout()
>>> sl.distance = 55
>>> sl.isNew = True
>>> sl.rightMargin = 30.25
>>> sl.leftMargin = 20

>>> XPBase = musicxml.m21ToXml.XMLExporterBase()
>>> mxPrint = XPBase.systemLayoutToXmlPrint(sl)
>>> XPBase.dump(mxPrint)
<print new-system="yes">
  <system-layout>
    <system-margins>
      <left-margin>20</left-margin>
      <right-margin>30.25</right-margin>
    </system-margins>
    <system-distance>55</system-distance>
  </system-layout>
</print>


Test return conversion

>>> MP = musicxml.xmlToM21.MeasureParser()
>>> sl2 = MP.xmlPrintToSystemLayout(mxPrint)
>>> sl2.isNew
True
>>> sl2.rightMargin
30.25
>>> sl2.leftMargin
20
>>> sl2.distance
55
Nr  
new-systemr  r  system-layout)r   r  r*   r8  systemLayoutToXmlSystemLayoutr  r   )ry   systemLayoutr  r  r  mxSystemLayouts         r9   systemLayoutToXmlPrint&XMLExporterBase.systemLayoutToXmlPrintX  sr    L g&GG(\L'ZE^E^_ 1**<H~NN>*N r;   c                    Uc  [        S5      nOUn[        n[        S5      nS H  nU" XUS-   5        M     [        U5      (       a  UR                  U5        U" XSS5        U" XSS	5        Uc  U$ g)
a  
get given a SystemLayout object configure <system-layout> or <print>

Called out from xmlPrintToSystemLayout because it
is also used in the <defaults> tag

>>> sl = layout.SystemLayout()
>>> sl.distance = 40.0
>>> sl.topDistance = 70.0
>>> XPBase = musicxml.m21ToXml.XMLExporterBase()
>>> mxSl = XPBase.systemLayoutToXmlSystemLayout(sl)
>>> XPBase.dump(mxSl)
<system-layout>
  <system-distance>40.0</system-distance>
  <top-system-distance>70.0</top-system-distance>
</system-layout>

>>> sl = layout.SystemLayout()
>>> sl.leftMargin = 30.0
>>> mxSl = XPBase.systemLayoutToXmlSystemLayout(sl)
>>> XPBase.dump(mxSl)
<system-layout>
  <system-margins>
    <left-margin>30.0</left-margin>
  </system-margins>
</system-layout>
Nr  zsystem-margins)r  r  r  r  r  zsystem-distancedistanceztop-system-distancetopDistancer  )ry   r  mxSystemLayoutInr  r  mxSystemMarginsr  s          r9   r  -XMLExporterBase.systemLayoutToXmlSystemLayout  s    8 #$_5N-N' ""23;I	I0EF < !!/2\+<jI\+@-P #!! $r;   c                p    Uc  [        S5      nOUn[        n[        nU" XSS5        U" XSS5        Uc  U$ g)a  
get a <staff-layout> tag from a StaffLayout object

In music21, the <staff-layout> and <staff-details> are
intertwined in a StaffLayout object.

>>> sl = layout.StaffLayout()
>>> sl.distance = 40.0
>>> sl.staffNumber = 1
>>> XPBase = musicxml.m21ToXml.XMLExporterBase()
>>> mxSl = XPBase.staffLayoutToXmlStaffLayout(sl)
>>> XPBase.dump(mxSl)
<staff-layout number="1">
  <staff-distance>40.0</staff-distance>
</staff-layout>

Nzstaff-layoutzstaff-distancer  r   staffNumber)r   rX   r  )ry   staffLayoutmxStaffLayoutInmxStaffLayoutr  r  s         r9   staffLayoutToXmlStaffLayout+XMLExporterBase.staffLayoutToXmlStaffLayout  sM    $ "#N3M+M'([)9:F[=A"   #r;   
accidentalelNamec                  SnUR                   S:X  a  SnOsUR                   S:X  a  SnO`UR                   S:X  a  SnOMUR                   S:X  a  S	nO:UR                   S
:X  a  SnO'UR                   nU[        R                  ;  a  XC;  a  Sn[        U5      nXEl        UR
                  S;   a  UR                  SS5        UR
                  S;   a  UR                  SS5        U R                  XQ5        U$ )a  
Convert a pitch.Accidental object to a Element of tag 'accidental' (or other name)

>>> acc = pitch.Accidental()
>>> acc.set('half-sharp')
>>> acc.alter == 0.5
True

>>> XB = musicxml.m21ToXml.XMLExporterBase()
>>> a2m = XB.accidentalToMx
>>> XB.dump(a2m(acc))
<accidental>quarter-sharp</accidental>

>>> acc.set('double-flat')
>>> XB.dump(a2m(acc))
<accidental>flat-flat</accidental>


>>> acc.set('one-and-a-half-sharp')
>>> XB.dump(a2m(acc, elName='accidental-mark'))
<accidental-mark>three-quarters-sharp</accidental-mark>

>>> acc.set('half-flat')
>>> XB.dump(a2m(acc))
<accidental>quarter-flat</accidental>

>>> acc.set('one-and-a-half-flat')
>>> XB.dump(a2m(acc))
<accidental>three-quarters-flat</accidental>

>>> acc.set('sharp')
>>> acc.displayStyle = 'parentheses'
>>> XB.dump(a2m(acc))
<accidental parentheses="yes">sharp</accidental>

>>> acc.displayStyle = 'bracket'
>>> XB.dump(a2m(acc))
<accidental bracket="yes">sharp</accidental>

>>> acc.displayStyle = 'both'
>>> XB.dump(a2m(acc))
<accidental bracket="yes" parentheses="yes">sharp</accidental>

>>> acc = pitch.Accidental('flat')
>>> acc.style.relativeX = -2
>>> XB.dump(a2m(acc))
<accidental relative-x="-2">flat</accidental>

>>> acc = pitch.Accidental()
>>> acc.name = 'double-sharp-down'  # musicxml 3.1
>>> XB.dump(a2m(acc))
<accidental>double-sharp-down</accidental>

>>> acc.name = 'funnyAccidental'  # unknown
>>> XB.dump(a2m(acc))
<accidental>other</accidental>
)zdouble-sharp-downzdouble-sharp-upzflat-flat-downzflat-flat-upz
arrow-downzarrow-upotherz
sharp-downzsharp-upznatural-downz
natural-upz	flat-downzflat-upzslash-quarter-sharpzslash-sharpz
slash-flatzdouble-slash-flatzsharp-1zsharp-2zsharp-3zsharp-5zflat-1zflat-2zflat-3zflat-4sorikoronz
half-sharpzquarter-sharpzone-and-a-half-sharpzthree-quarters-sharpz	half-flatzquarter-flatzone-and-a-half-flatzthree-quarters-flatzdouble-flatz	flat-flatr  )parenthesesbothr  yes)bracketr  r  )r   r   accidentalNameToModifierr   r&   displayStyler:  rX  )ry   ar  otherMusicXMLAccidentalsmxNamemxAccidentals         r9   accidentalToMxXMLExporterBase.accidentalToMx  s    v$
 ( 66\!$FVV--+FVV{"#FVV,,*FVV}$ FVVFe<<<> v #>>44]E2>>00Y.<+r;   c                "    U R                  USS9$ )Nzaccidental-markr  )r  )ry   r  s     r9   accidentalToMxAccidentalMark,XMLExporterBase.accidentalToMxAccidentalMarkN  s    ""1->"??r;   )r"   r  r   None)T)r   r   )r=   )r  rR   r   r  r   )r  pitch.Accidentalr  rR   r   r   )r  r  r   r   )r   r   r   r   r   r{   r   r%  staticmethodr(  r  rC  rQ  rO  rX  rd  r^  r]  r\  r  r  r  r  r  r  r  r  r   r   r;   r9   r  r    s    5%: ! !I&,P5$F	+/	P7;rN?OL5n 64l1"f!@ DP jZ@r;   r  c                    ^  \ rS rSrSrSSU 4S jjjrS S jrS S jrS!S jrS!S jr	S!S jr
S!S	 jrS!S
 jrS!S jrS!S jrS!S jrS!S jrS!S jrS!S jrS"S jrS rS rS rS rS rS S jr S#   S$S jjrS rS rS!S jrS rSrU =r $ )%r   iT  zH
Convert a Score (or outer stream with .parts) into
a musicxml Element.
c                  > [         TU ]  5         Uc  [        R                  " 5       U l        OXl        [	        S[
        R                  S9U l        S U l        S U l	        S U l
        S U l        / U l        S U l        / U l        SU l        SU R                  /U l        / U l        / U l        0 U l        / U l        / U l        / U l        / U l        X l        g )Nzscore-partwise)versionr   )superr{   r"   r\   r   r   r+  r  mxIdentificationscoreMetadataspannerBundlemeterStreamscoreLayoutsfirstScoreLayout	textBoxesr   r   partExporterListgroupsToJoininstrumentsByStreaminstrumentListinstrumentIdListmidiChannelListpartsrx   )ry   scorerx   	__class__s      r9   r{   ScoreExporter.__init__Z  s    = ,,.DKK/9Q9QR.2599=HL689=-/%($*:*:$;!4657=? ;=02/1(*
".r;   c                   U R                   nU(       d  U R                  5       $ UR                  SSS9  U R                  5         UR	                  5       (       aV  U R                  5         U R                  5       U l        U R                  5         U R                  5         U R                  5         OU R                  5         U R                  5         U R                  R                  5         U R                  $ )a  
the main function to call.

If self.stream is empty, call self.emptyObject().  Otherwise,
convert sounding to written pitch,
set scorePreliminaries(), call parsePartlikeScore or parseFlatScore, then postPartProcess(),
clean up circular references for garbage collection, and returns the <score-partwise>
object.

>>> b = corpus.parse('bwv66.6')
>>> SX = musicxml.m21ToXml.ScoreExporter(b)
>>> mxScore = SX.parse()
>>> SX.dump(mxScore)
<score-partwise version="...">...</score-partwise>
Tr   ottavasToSounding)r"   emptyObjecttoWrittenPitchscorePreliminariesr   _populatePartExporterListjoinableGroupsr  setPartExporterStaffGroupsrenumberVoicesWithinStaffGroupsparsePartlikeScoreparseFlatScorepostPartProcessr  clearr  )ry   rG   s     r9   r   ScoreExporter.parse~  s      KK##%% 	
>!!! **, $ 3 3 5D++-002##%! 	##%||r;   c                   [         R                  " 5       n[         R                  " 5       n[         R                  " 5       n[        R
                  " SS9nUR                  U5        UR                  U5        UR                  U5        [        R                  " SS9nUR                  SU5        Xl         U R                  5       $ )a  
Creates a cheeky "This Page Intentionally Left Blank" for a blank score

>>> emptySX = musicxml.m21ToXml.ScoreExporter()
>>> mxScore = emptySX.parse()  # will call emptyObject
>>> emptySX.dump(mxScore)
<score-partwise version="...">
  <work>
    <work-title>This Page Intentionally Left Blank</work-title>
  </work>
  ...
      <note>
        <rest />
        <duration>40320</duration>
        <type>whole</type>
      </note>
    </measure>
  </part>
</score-partwise>
r   )r   z"This Page Intentionally Left Blank)titler   )r"   r\   r^   r`   r   Restr   r   Metadatar   r   )ry   r   r   r   rrH   s         r9   r  ScoreExporter.emptyObject  s    * llnKKMNNIIA&		

1%IJ

1bzz|r;   c                   U R                  5         U R                  5         U R                  5         [        U R                  [
        R                     5      U l        SU l        U R                  c  U R                  R                  U l	        gg)a0  
Populate the exporter object with
`meterStream`, `scoreLayouts`, `spannerBundle`, and `textBoxes`

>>> emptySX = musicxml.m21ToXml.ScoreExporter()
>>> emptySX.scorePreliminaries()  # will call emptyObject
>>> len(emptySX.textBoxes)
0
>>> emptySX.spannerBundle
<music21.spanner.SpannerBundle of size 0>

r   N)
setScoreLayoutssetMeterStreamsetPartsAndRefStreamlistr"   r&   TextBoxr  r   r  r  s    r9   r   ScoreExporter.scorePreliminaries  sn     	!!#dkk$,,78 %!%!:!:D &r;   c                   U R                   n[        R                  (       a!  [        U[         R                  5      (       d   eUR                  [         R                  5      nU H  n[        U[         R                  5      (       a  UR                  UR                  -   nOV[        U[         R                  [         R                  45      (       a  [        S5      eUR                  5         UR                  n[        U R                  U5      U l	        M     SU R                  /U l        [        UR                   5      U l        g)aK  
Transfers the offset of the inner stream to elements and sets self.highestTime

>>> b = corpus.parse('bwv66.6')
>>> SX = musicxml.m21ToXml.ScoreExporter(b)
>>> SX.highestTime
0.0
>>> SX.setPartsAndRefStream()
>>> SX.highestTime
36.0
>>> SX.refStreamOrTimeRange
[0.0, 36.0]
>>> len(SX.parts)
4
>>> isinstance(SX.parts[0], stream.Part)
True

>>> b.insert(stream.Score())
>>> SX = musicxml.m21ToXml.ScoreExporter(b)
>>> SX.setPartsAndRefStream()
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
Exporting scores nested inside scores is not supported
z6Exporting scores nested inside scores is not supportedr   N)r"   tTYPE_CHECKINGr   r\   r   rd   r`   offsetr   Opusr,   transferOffsetToElementsmaxr   r  r  )ry   rG   streamOfStreamsinnerStreamhts        r9   r  "ScoreExporter.setPartsAndRefStream  s    2 KK??a.... ..v}}=*K
 +v~~66 ''+*A*AAK&,,)DEE-LN N 446 ,,"4#3#3R8D + &)$*:*:$;!!'']
r;   c                J    U R                   nUR                  SSSS9nX l        g)aC  
sets `self.meterStream` or uses a default.

Used in makeNotation in Part later.

>>> b = corpus.parse('bwv66.6')
>>> SX = musicxml.m21ToXml.ScoreExporter(b)
>>> SX.setMeterStream()
>>> SX.meterStream
<music21.stream.Score bach/bwv66.6.mxl>
>>> len(SX.meterStream)
4
>>> SX.meterStream[0]
<music21.meter.TimeSignature 4/4>
FT)searchContextsortByCreationTimereturnDefaultN)r"   getTimeSignaturesr  )ry   rG   r  s      r9   r  ScoreExporter.setMeterStream  s2      KK))=B8< * > 'r;   c                    U R                   nUR                  [        R                  5      nUR	                  5       U l        [        U5      U l        g)a,  
sets `self.scoreLayouts` and `self.firstScoreLayout`

>>> b = corpus.parse('schoenberg/opus19', 2)
>>> SX = musicxml.m21ToXml.ScoreExporter(b)
>>> SX.setScoreLayouts()
>>> SX.scoreLayouts
[<music21.layout.ScoreLayout>]
>>> len(SX.scoreLayouts)
1
>>> SX.firstScoreLayout
<music21.layout.ScoreLayout>
N)r"   r   r   ScoreLayoutr   r  r  r  )ry   rG   r  s      r9   r  ScoreExporter.setScoreLayouts+  sA     KK++F,>,>? , 2 2 4 .r;   c                    Sn[        U R                  5      nU HW  nUS-  nU[        U5      :  a  [        S5      e[	        X0S9nU R
                  Ul        U R                  R                  U5        MY     g )Nr   r   zinfinite stream encounteredparent)r  r  r  r,   PartExporterr  r  r   )ry   countspr  pps        r9   r  'ScoreExporter._populatePartExporterList>  sm    $**KQJE s2w-.KLLk7B#11B!!((, r;   c                    U R                   (       d  U R                  5         U R                    H  nUR                  5         M     g)z
Called by .parse() if the score has individual parts.
Creates a `PartExporter` for each part, and runs .parse() on that part.
Appends the PartExporter to `self.partExporterList`
and runs .parse() on that part. Appends the PartExporter to self.
N)r  r  r   )ry   part_exs     r9   r   ScoreExporter.parsePartlikeScoreL  s2     $$**,,,GMMO -r;   c                   U R                   n[         R                  " 5       nU H  nUR                  UR                  U5        M!     UR	                  5         [        X S9nUR                  5         U R                  R                  U5        g)a  
creates a single PartExporter for this Stream and parses it.

Note that the Score does not need to be totally flat, it just cannot have Parts inside it;
measures are fine.

>>> c = converter.parse('tinyNotation: 3/4 c2. d e')
>>> SX = musicxml.m21ToXml.ScoreExporter(c)
>>> SX.parseFlatScore()
>>> len(SX.partExporterList)
1
>>> SX.partExporterList[0]
<music21.musicxml.m21ToXml.PartExporter object at 0x...>
>>> SX.dump(SX.partExporterList[0].xmlRoot)
<part id="...">
  <!--========================= Measure 1 ==========================-->
  <measure implicit="no" number="1">...</measure>
</part>
>>> del SX.partExporterList[:]  # for garbage collection
r  N)	r"   r^   
coreInsertr  coreElementsChangedr!  r   r  r   )ry   rG   r   elr$  s        r9   r  ScoreExporter.parseFlatScoreX  sg    * KKKKMBLLB' 	!)

$$R(r;   c                    U R                    H2  nSnU R                   H  nUR                  U;   d  M  Un  O   X!l        M4     g)zn
Figures out the containing StaffGroup for every PartExporter that has one.

Called automatically by .parse()
N)r  r  r"   
staffGroup)ry   partExpjoinableGroupsgs       r9   r  (ScoreExporter.setPartExporterStaffGroupsv  sE     ,,G M''>>R'$&M (
 "/ -r;   c                   / nU R                    GH=  nUR                  c  M  UR                  U;   a  M%  [        R                  " UR                  R	                  5       5      nUR                  5       R                  [        R                  5      R                  5       n[        U5      nU H  nSnU Hx  nU[        R                      H^  n	[        U	R                  [        5      (       d  M$  U	R                  [        R                  :  a  U	R                  S-   nMS  Xyl        US-  nM`     Mz     M     UR                  UR                  5        GM@     g)z
Renumbers voices (as appropriate) in each StaffGroup, so that
voices have unique numbers across the entire group.

Called automatically by .parse()
Nr   )r  r/  r"   r\   getSpannedElementsr   r   r`   r#   rb   r   idintr   #minIdNumberToConsiderMemoryLocationr   )
ry   staffGroupsProcessedr0  staffGroupScoremeasuresStreamoffsetIteratormeasureStacknextVoiceIdr   r   s
             r9   r  -ScoreExporter.renumberVoicesWithinStaffGroups  s    &(,,G!!)!!%99 %ll7+=+=+P+P+RSO'')<<V^^LSSU  >LN=[N .#$%Av||_)!$$44$TTH$P$PP*+$$(K $/D'1,K - & !/  !''(:(:;A -r;   c                
   U R                  5         U R                  5         [        U R                  5       HJ  u  pU R	                  S[        US-   5      -   5        U R                  R                  UR                  5        ML     g)z
calls .joinPartStaffs() from the
:class:`~music21.musicxml.partStaffExporter.PartStaffExporterMixin`,
then calls .setScoreHeader(),
then appends each PartExporter's xmlRoot from
self.partExporterList to self.xmlRoot.

Called automatically by .parse().
zPart r   N)joinPartStaffssetScoreHeader	enumerater  r%  rR   r  r   )ry   r   pexs      r9   r  ScoreExporter.postPartProcess  sd     	 5 56FA""7SQZ#78LL, 7r;   c                6   U R                   n[        U5      U l        U R                  5         U R	                  5         U R                  5         U R                   H-  nU R                  R                  U R                  U5      5        M/     U R                  5         g)a8  
Sets the group score-header in <score-partwise>.  Note that score-header is not
a separate tag, but just a way of grouping things from the tag.

runs `setTitles()`, `setIdentification()`, `setDefaults()`, changes textBoxes
to `<credit>` and does the major task of setting up the part-list with `setPartList()`.
N)r"   rJ   r  	setTitlessetIdentificationsetDefaultsr  r  r   textBoxToXmlCreditsetPartList)ry   rG   tbs      r9   rB  ScoreExporter.setScoreHeader  sw     KK 4A6 ..BLL 7 7 ;< ! 	r;   c                ^   [        S5      nUR                  b&  UR                  S[        UR                  5      5        OUR                  SS5        [        S5      nSUR                  ;   a  UR                  SS5        UR                  Ul        U R                  X15        UR                  (       ak  UR                  n[        R                  (       a!  [        U[        R                  5      (       d   eUR                  b  UR                  SUR                  5        UR                  U5        U$ )	a  
Convert a music21 TextBox to a MusicXML Credit.

>>> tb = text.TextBox('testing')
>>> tb.style.absoluteY = 400
>>> tb.style.absoluteX = 300
>>> tb.page = 3
>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> mxCredit = SX.textBoxToXmlCredit(tb)
>>> SX.dump(mxCredit)
<credit page="3">
  <credit-words default-x="300" default-y="400"
       halign="center" valign="top">testing</credit-words>
</credit>

Default of page 1:

>>> tb = text.TextBox('testing')
>>> tb.page
1
>>> mxCredit = SX.textBoxToXmlCredit(tb)
>>> SX.dump(mxCredit)
<credit page="1">...</credit>

* Changed in v8: Multi-line text now exports as one `<credit-words>`
  element (preserving newlines).

>>> tb = text.TextBox('Snare\nCymbals')
>>> mxCredit = SX.textBoxToXmlCredit(tb)
>>> SX.dump(mxCredit)
<credit page="1">
    <credit-words default-x="500" default-y="500" halign="center" valign="top"
    xml:space="preserve">Snare
    Cymbals</credit-words>
</credit>
creditpage1zcredit-words
z	xml:spacepreserverF  )r   rP  r:  rR   contentr&   rO  r4  r$   r  r  r   	TextStylerF  r   )ry   textBoxmxCreditmxCreditWordsstys        r9   rJ   ScoreExporter.textBoxToXmlCredit  s    L 8$ <<#LLW\\!23LL%/7??"k:6$__7&&--C!#u7777{{&!!)S[[9&r;   c                8   U R                   c@  [        R                  " 5       n[        R                  Ul        [        R
                  Ul        OU R                   n[        U R                  S5      nUR                  c  UR
                  bX  [        US5      n[        US5      n[        UR                  5      Ul	        [        US5      n[        UR
                  5      Ul	        UR                  b,  U R                  UR                  5      nUR                  U5        UR                  b,  U R                  UR                  5      nUR                  U5        UR                   H%  nU R!                  U5      n	UR                  U	5        M'     U R#                  U5        U$ )av  
Returns a default object from self.firstScoreLayout or a very simple one if none exists.

Simple:

>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> mxDefaults = SX.setDefaults()
>>> SX.dump(mxDefaults)
<defaults>
  <scaling>
    <millimeters>7</millimeters>
    <tenths>40</tenths>
  </scaling>
</defaults>

These numbers come from the `defaults` module:

>>> defaults.scalingMillimeters
7
>>> defaults.scalingTenths
40

More complex:

>>> s = corpus.parse('schoenberg/opus19', 2)
>>> SX = musicxml.m21ToXml.ScoreExporter(s)
>>> SX.setScoreLayouts()  # necessary to call before .setDefaults()
>>> mxDefaults = SX.setDefaults()
>>> mxDefaults.tag
'defaults'
>>> mxScaling = mxDefaults.find('scaling')
>>> SX.dump(mxScaling)
<scaling>
  <millimeters>6.1472</millimeters>
  <tenths>40</tenths>
</scaling>

>>> mxPageLayout = mxDefaults.find('page-layout')
>>> SX.dump(mxPageLayout)
<page-layout>
  <page-height>1818</page-height>
  <page-width>1405</page-width>
  <page-margins>
    <left-margin>83</left-margin>
    <right-margin>83</right-margin>
    <top-margin>103</top-margin>
    <bottom-margin>103</bottom-margin>
  </page-margins>
</page-layout>

>>> mxSystemLayout = mxDefaults.find('system-layout')
>>> SX.dump(mxSystemLayout)
<system-layout>
  <system-margins>
    <left-margin>0</left-margin>
    <right-margin>0</right-margin>
  </system-margins>
  <system-distance>121</system-distance>
  <top-system-distance>70</top-system-distance>
</system-layout>

>>> mxStaffLayoutList = mxDefaults.findall('staff-layout')
>>> len(mxStaffLayoutList)
1
>>> SX.dump(mxStaffLayoutList[0])
<staff-layout>
  <staff-distance>98</staff-distance>
</staff-layout>
r   scalingmillimeterstenths)r  r   r  r   scalingMillimetersscalingTenthsr   r  rR   r&   r  r  r   r  r  staffLayoutListr  addStyleToXmlDefaults)
ry   scoreLayout
mxDefaults	mxScalingmxMillimetersmxTenthsr  r  r  r  s
             r9   rI  ScoreExporter.setDefaults  sZ   R   ( ,,.K-5-H-HK*(0(>(>K%//Kj9
))59R9R9^":y9I&y-@M!$[%C%C!DM!)X6H 9 9:HM!!-99+:P:PQLl+##/!??@X@XYNn-&66K <<[IMm, 7 	"":.r;   c                   U R                   R                  (       d  gU R                  5       nUb  UR                  U5        U R                   R                  nUR
                  b'  [        US5      nU R                  XCR
                  5        UR                  b'  [        US5      nU R                  XSR                  5        UR                   H<  u  pg[        US5      nUb  U R                  X5        Uc  M*  UR                  SU5        M>     UR                   H=  u  p[        US5      nU	b  UR                  SU	5        U
c  M+  UR                  SU
5        M?     g)a"  
Optionally add an <appearance> tag (using `styleToXmlAppearance`)
and <music-font>, <word-font>, zero or more <lyric-font> tags,
and zero or more <lyric-language> tags to mxDefaults

Demonstrating round tripping:

>>> import xml.etree.ElementTree as ET
>>> defaults = ET.fromstring('<defaults>'
...          + '<music-font font-family="Maestro, Opus" font-weight="bold" />'
...          + '<word-font font-family="Garamond" font-style="italic" />'
...          + '<lyric-font name="verse" font-size="12" />'
...          + '<lyric-font name="chorus" font-size="14" />'
...          + '<lyric-language name="verse" xml:lang="fr" />'
...          + '<lyric-language name="chorus" xml:lang="en" />'
...          + '</defaults>')

>>> MI = musicxml.xmlToM21.MusicXMLImporter()
>>> MI.styleFromXmlDefaults(defaults)
>>> SX = musicxml.m21ToXml.ScoreExporter(MI.stream)
>>> mxDefaults = ET.Element('defaults')
>>> SX.addStyleToXmlDefaults(mxDefaults)
>>> SX.dump(mxDefaults)
<defaults>
    <music-font font-family="Maestro,Opus" font-weight="bold" />
    <word-font font-family="Garamond" font-style="italic" />
    <lyric-font font-size="12" name="verse" />
    <lyric-font font-size="14" name="chorus" />
    <lyric-language name="verse" xml:lang="fr" />
    <lyric-language name="chorus" xml:lang="en" />
</defaults>
Nz
music-fontz	word-fontz
lyric-fontr   zlyric-languagezxml:lang)r"   r4  styleToXmlAppearancer   r$   	musicFontr   r]  wordFont
lyricFontsr:  lyricLanguages)ry   rd  mxAppearancer   mxMusicFont
mxWordFont	lyricName	lyricFontmxLyricFont	lyricType	lyricLangmxLyricLanguages               r9   rb  #ScoreExporter.addStyleToXmlDefaults~  s#   D {{..002#l+[[<<#$Z>KLLll3;;"#J<JLL[[1$&MM I$Z>K$[4$	2 %2 %'$5$5 I(5EFO$##FI6$##J	: %6r;   c                n   U R                   R                  nUR                  (       d4  UR                  (       d#  UR                  (       d  UR
                  (       d  g[        S5      nS HH  u  p4[        X5      nU H2  u  pg[        X$5      nUR                  SU5        [        U5      Ul        M4     MJ     U$ )a  
Populates the <appearance> tag of the <defaults> with
information from the stream's .style information.

>>> s = stream.Score()
>>> s.style.lineWidths.append(('beam', 5.0))
>>> s.style.noteSizes.append(('cue', 75))
>>> s.style.distances.append(('hyphen', 0.1))
>>> s.style.otherAppearances.append(('flags', 'wavy'))
>>> SX = musicxml.m21ToXml.ScoreExporter(s)
>>> mxAppearance = SX.styleToXmlAppearance()
>>> SX.dump(mxAppearance)
<appearance>
  <line-width type="beam">5.0</line-width>
  <note-size type="cue">75</note-size>
  <distance type="hyphen">0.1</distance>
  <other-appearance type="flags">wavy</other-appearance>
</appearance>
N
appearance))
lineWidthsz
line-width)	noteSizesz	note-size)	distancesr  )otherAppearanceszother-appearancetype)r"   r$   r{  r|  r}  r~  r   rP   r   r:  rR   r&   )	ry   r   ro  thisPropertyrU   propertyListpropertyTypepropertyValue
mxPropertys	            r9   rj  "ScoreExporter.styleToXmlAppearance  s    * [[++|,"LL #24L/;+':
v|4"%m"4
 0<"L r;   c                F   U R                   n[        U R                  S5      nUR                  [        R
                  5      n0 nSnU R                   GHJ  nUR                  nU Hr  nXR                  ;   a  M  UR                  U5      (       d  M,  U R                  U5      n	U	R                  S[        U5      5        XU'   US-  nUR                  U	5        Mt     UR                  5       n
UR                  U
5        SnU H  nXR                  ;   a  M  UR                  U5      (       d  M,  UR!                  5        H  u  pXL d  M  Un  O   [#        S5      n	U	R                  SS5        Ub  U	R                  S[        U5      5        UR                  U	5        M     GMM     U$ )a  
Returns a <part-list> and appends it to self.xmlRoot.

This is harder than it looks because MusicXML and music21's idea of where to store
staff-groups are quite different.

We find each stream in self.partExporterList, then look at the StaffGroup spanners in
self.spannerBundle.  If the part is the first element in a StaffGroup then we add a
<staff-group> object with 'start' as the starting point (and same for multiple StaffGroups)
this is in `staffGroupToXmlPartGroup(sg)`.
then we add the <score-part> descriptor of the part and its instruments, etc. (currently
just one!), then we iterate again through all StaffGroups and if this part is the last
element in a StaffGroup we add a <staff-group> descriptor with type="stop".

This Bach example has four parts and one staff-group bracket linking them:

>>> b = corpus.parse('bwv66.6')
>>> SX = musicxml.m21ToXml.ScoreExporter(b)

Needs some strange setup to make this work in a demo.  `.parse()` takes care of all this.

>>> SX.scorePreliminaries()
>>> SX.parsePartlikeScore()

>>> mxPartList = SX.setPartList()
>>> SX.dump(mxPartList)
<part-list>
  <part-group number="1" type="start">...
  <score-part id="P1">...
  <score-part id="P2">...
  <score-part id="P3">...
  <score-part id="P4">...
  <part-group number="1" type="stop" />
</part-list>

Multi-staff parts (such as piano staves), should NOT receive `<part-group>` tags,
since they are joined by `<staff>` tags:

>>> cpe = corpus.parse('cpebach')
>>> SX = musicxml.m21ToXml.ScoreExporter(cpe)
>>> SX.scorePreliminaries()
>>> SX.parsePartlikeScore()
>>> SX.joinPartStaffs()

>>> mxPartList = SX.setPartList()
>>> SX.dump(mxPartList)
<part-list>
  <score-part id="P1">...
  </score-part>
</part-list>
z	part-listr   r   N
part-groupr  stop)r  r   r  
getByClassr   
StaffGroupr  r"   r  isFirststaffGroupToXmlPartGroupr:  rR   r   getXmlScorePartisLastr   r   )ry   r  
mxPartListstaffGroupspartGroupIndexRefpartGroupIndexrD  r   r2  mxPartGroupmxScorePartactiveIndexkr8   s                 r9   rK  ScoreExporter.setPartList  s|   l **k:
 $..v/@/@A ((C

A!***::a=="&"?"?"CKOOHc..AB8:n5"a'N%%k2 " --/Kk*K!***99Q<<$5$;$;$= ;*+K! %> #*,"7KOOFF3".##k2BC%%k2 "% )D r;   c                   [        S5      nUR                  SS5        [        nU" XSS5        UR                  R                  (       a  [        US5      nUR                  SS5        U" XS	S
5        U" XSS5      nUb"  U R                  XQ5        U R                  XQ5        [        US5      nUR                  SL a  SUl	        O.UR                  SL a  SUl	        OUR                  S:X  a  SUl	        U R                  X!5        U$ )aT  
Create and configure an mxPartGroup object for the 'start' tag
from a staff group spanner. Note that this object
is not completely formed by this procedure (number isn't done).

>>> b = corpus.parse('bwv66.6')
>>> SX = musicxml.m21ToXml.ScoreExporter(b)
>>> firstStaffGroup = b.spannerBundle.getByClass(layout.StaffGroup)[0]
>>> mxPartGroup = SX.staffGroupToXmlPartGroup(firstStaffGroup)
>>> SX.dump(mxPartGroup)
<part-group type="start">
  <group-symbol>bracket</group-symbol>
  <group-barline>yes</group-barline>
</part-group>

At this point, you should set the number of the mxPartGroup, since it is required:

>>> mxPartGroup.set('number', str(1))


What can we do with it?

>>> firstStaffGroup.name = 'Voices'
>>> firstStaffGroup.abbreviation = 'Ch.'
>>> firstStaffGroup.symbol = 'brace' # 'none', 'brace', 'line', 'bracket', 'square'
>>> firstStaffGroup.barTogether = False  # True, False, or 'Mensurstrich'
>>> mxPartGroup = SX.staffGroupToXmlPartGroup(firstStaffGroup)
>>> SX.dump(mxPartGroup)
<part-group type="start">
  <group-name>Voices</group-name>
  <group-abbreviation>Ch.</group-abbreviation>
  <group-symbol>brace</group-symbol>
  <group-barline>no</group-barline>
</part-group>

Now we avoid printing the name of the group:

>>> firstStaffGroup.style.hideObjectOnPrint = True
>>> mxPartGroup = SX.staffGroupToXmlPartGroup(firstStaffGroup)
>>> SX.dump(mxPartGroup)
<part-group type="start">
  <group-name>Voices</group-name>
  <group-name-display print-object="no" />
  <group-abbreviation>Ch.</group-abbreviation>
  <group-symbol>brace</group-symbol>
  <group-barline>no</group-barline>
</part-group>
r  r  startz
group-namer   zgroup-name-displayra  rb  zgroup-abbreviationabbreviationzgroup-symbolsymbolzgroup-barlineTr  FMensurstrich)r   r:  rX   r$   rc  r   r^  r\  barTogetherr&   r  )ry   r/  r  r  mxGroupNameDisplaymxGroupSymbolmxGroupBarlines          r9   r  &ScoreExporter.staffGroupToXmlPartGroupM  s    d l+('ZlF;--!+K9M!N"">48Z&:NKZnhO$MM-4]7#KA!!T)"'N##u,"&N##~5"0N+2 r;   c                   U R                   b  U R                   nO[        U R                  S5      nXl         SnU R                  bH  U R                  R	                  SSSS9 H)  u  p4U R                  U5      nUR                  U5        SnM+     USL aH  [        R                  (       a3  [        US5      nUR                  SS5        [        R                  Ul
        U R                  b]  U R                  S   nU HH  n[        US	5      nUR                  b  UR                  SUR                  5        [        U5      Ul
        MJ     U R                  5         U R                  5         U$ )
a  
Returns an identification object from self.scoreMetadata and appends it to the score.

For defaults:

>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> mxIdentification = SX.setIdentification()
>>> SX.dump(mxIdentification)
<identification>
  <creator type="composer">Music21</creator>
  <encoding>
    <encoding-date>20...-...-...</encoding-date>
    <software>music21 v...</software>
  </encoding>
</identification>

More realistic:

>>> md = metadata.Metadata()
>>> md.composer = 'Francesca Caccini'
>>> c = metadata.Contributor(role='arranger', name='Aliyah Shanti')
>>> md.addContributor(c)

Need a fresh ScoreExporter, or, it will otherwise append to the existing mxIdentification

>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> SX.scoreMetadata = md
>>> mxIdentification = SX.setIdentification()
>>> SX.dump(mxIdentification)
<identification>
  <creator type="composer">Francesca Caccini</creator>
  <creator type="arranger">Aliyah Shanti</creator>
  <encoding>
    <encoding-date>...</encoding-date>
    <software>music21 v...</software>
  </encoding>
</identification>


Overriding the default:

>>> defaults.author = "Batch Conversion March 2022"
>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> mxIdentification = SX.setIdentification()
>>> SX.dump(mxIdentification)
<identification>
  <creator type="composer">Batch Conversion March 2022</creator>
  <encoding>
    <encoding-date>20...-...-...</encoding-date>
    <software>music21 v...</software>
  </encoding>
</identification>

identificationFT)skipNonContributorsreturnPrimitivesreturnSortedcreatorr  composer	copyrightrights)r  r   r  r  allcontributorToXmlCreatorr   r   authorr:  r&   rolerR   setEncodingmetadataToMiscellaneous)ry   mxIdfoundOne_r  	mxCreator
copyrightsmxRightss           r9   rH  ScoreExporter.setIdentification  sE   p   ,((Ddll,<=D$(! )
 **..(,%)!& / ( !88;	I&( u"43IMM&*-%__IN)9=9K9KK9XJ%dH566%LL0 #A	   	 	$$&r;   c                x   Uc  U R                   c  gUc  U R                   n[        R                  (       a  Uc   e[        S5      nSnUR	                  SSSS9nSnSnSnU H  u  pUS:X  a  M  US:X  a  U(       d  SnM  US:X  a  U(       d  SnM/  US	:X  a  U(       d  SnM@  US
:X  a  MH  UR                  U5      n
U
c  Un
U
R                  S5      (       a  Mv  [        US5      nUR                  SU
5        [        U	5      Ul
        SnM     U R                  b"  U(       a  U R                  R                  U5        U$ )a  
Returns an mxMiscellaneous of information from metadata object md or
from self.scoreMetadata if md is None.  If the mxMiscellaneous object
has any miscellaneous-fields, then it is appended to self.mxIdentification
if it exists.

>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> md = metadata.Metadata()
>>> md.date = metadata.primitives.DateRelative('1689', 'onOrBefore')
>>> md.localeOfComposition = 'Rome'

>>> mxMisc = SX.metadataToMiscellaneous(md)
>>> SX.dump(mxMisc)
<miscellaneous>
  <miscellaneous-field name="dcterms:created">1689/--/-- or earlier</miscellaneous-field>
  <miscellaneous-field name="humdrum:OPC">Rome</miscellaneous-field>
</miscellaneous>
NmiscellaneousFT)skipContributorsr  r  softwaremovementNamemovementNumberr  r  zm21FileInfo:zmiscellaneous-fieldr   )r  r  r  r   r  uniqueNameToNamespaceName
startswithr   r:  rR   r&   r  r   )ry   rH   mxMiscellaneousr  allItemsskippedOneMovementNameskippedOneMovementNumberskippedOneTitle
uniqueNamer8   namespaceNamemxMiscFields               r9   r  %ScoreExporter.metadataToMiscellaneous  sf   . :$,,4Z##B??>!>!/266!!  
 (-).  %!)JZ'^+ .-1*-- 0/3,W$ '&*O[(&(&B&B:&NM$ *''77 $_6KLKOOFM2"5zKHW "*Z   ,!!((9 r;   c                
   U R                   b  [        U R                   S5      nO[        S5      n[        US5      n[        [        R
                  R                  5       5      Ul        U R                  bC  SnU R                  R                   H&  nSU;   a  U(       a  M  Sn[        US5      nXEl        M(     O![        US5      n[        R                  Ul        U R                  5       nU H  nUR                  U5        M     U$ )a  
Returns an encoding object that might have information about <supports> also.
and appends to mxIdentification (if any)

Will use the date of generation as encoding-date.

>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> mxEncoding = SX.setEncoding()
>>> SX.dump(mxEncoding)
<encoding>
  <encoding-date>20...-...-...</encoding-date>
  <software>music21 v...</software>
</encoding>

Encoding-date is in YYYY-MM-DD format.
encodingzencoding-dateFz
music21 v.Tr  )r  r   r   rR   datetimedatetodayr&   r  r  r   getSupportsr   )ry   
mxEncodingmxEncodingDatefound_m21_alreadyr  
mxSoftwaremxSupportsList
mxSupportss           r9   r  ScoreExporter.setEncodingZ	  s    $   ,#D$9$9:FJ ,J#J@!(--"5"5"78 ) % ..778+( !,0)'
J?
"* 8 $J
;J&//JO ))+(Jj) ) r;   c                    S n/ nU R                   nUR                  SL a  UR                  U" SSSS5      5        UR                  SL a  UR                  U" SSSS5      5        U$ )ax  
return a list of <supports> tags  for what this supports.  Does not append

Currently just supports new-system and new-page if s.definesExplicitSystemBreaks
and s.definesExplicitPageBreaks is True.

>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> SX.getSupports()
[]
>>> SX.stream.definesExplicitSystemBreaks = True
>>> SX.getSupports()
[<Element 'supports' at 0x...>]
>>> SX.dump(SX.getSupports()[0])
<supports attribute="new-system" element="print" type="yes" value="yes" />

>>> SX.stream.definesExplicitPageBreaks = True
>>> SX.dump(SX.getSupports()[1])
<supports attribute="new-page" element="print" type="yes" value="yes" />

c                    [        S5      nUR                  SU 5        UR                  SU5        UR                  SU5        UR                  SU5        U$ )Nsupports	attributer  r8   element)r   r:  )r  supports_typer8   r  sus        r9   
getSupport-ScoreExporter.getSupports.<locals>.getSupport	  sI    $BFF;	*FF6=)FF7E"FF9g&Ir;   Tr  r  r  r  )r"   definesExplicitSystemBreaksr   definesExplicitPageBreaks)ry   r  supportsListrG   s       r9   r  ScoreExporter.getSupports	  sk    *	 KK((D0
<w OP&&$.
:ueW MNr;   c                   U R                   nU R                   c  [        R                  " 5       n[        R                  (       a  Uc   eU R
                  n[        S5      nSnUS   nU(       a'  Uc  US   n[        US5      n[        US   5      Ul	        [        U5      (       a  UR                  U5        US   nU(       a  [        US5      n[        US   5      Ul	        Sn	US	   n
U
(       a  [        U
S   5      n	O6Ub  [        U5      n	O'[        R                  (       a  [        R                  n	Og[        US
5      nXl	        g)zT
puts work (with work-title), movement-number, movement-title into the self.xmlRoot
Nworkr  r   z
work-titler  zmovement-numberr=   r  zmovement-title)r  r   r   r  r  r  r   r   rR   r&   r  r   r   r  )ry   mdObjmxScoreHeadermxWorkfirstTitleFoundtitlesmxWorkTitlemovementNumbersmxMovementNumbermovement_titlemovementNamesmxMovementTitles               r9   rG  ScoreExporter.setTitles	  s)    ""%%%'E??$$$.2,1'N&"()$V\:K"6!9~Kv;;  (5:;K5L)-9JK$'(:$;! !383H q!12N*!$_!5!)$]4DE-r;   c                    [        S5      nUR                  b%  UR                  S[        UR                  5      5        UR                  b  UR                  Ul        U$ )a  
Return a <creator> tag from a :class:`~music21.metadata.Contributor` object.

>>> md = metadata.Metadata()
>>> md.composer = 'Oliveros, Pauline'
>>> contrib = md.contributors[0]
>>> contrib
<music21.metadata.primitives.Contributor composer:Oliveros, Pauline>

>>> SX = musicxml.m21ToXml.ScoreExporter()
>>> mxCreator = SX.contributorToXmlCreator(contrib)
>>> SX.dump(mxCreator)
<creator type="composer">Oliveros, Pauline</creator>
r  r  )r   r  r:  rR   r   r&   )ry   r  r  s      r9   r  %ScoreExporter.contributorToXmlCreator	  sI      I&	66MM&#aff+.66VVINr;   )r  r  r   r  r  r  rx   r  r  r  r  r  r   r  r  r  r"   r  r  ru   )r  zstream.Score | Nonerx   boolr   r   r  )rV  ztext.TextBoxr   r   r   )rH   metadata.Metadata | Noner   Element | None)!r   r   r   r   r   r{   r   r  r  r  r  r  r  r  r  r  r  r  rB  rJ  rI  rb  rj  rK  r  rH  r  r  r  rG  r  r   __classcell__r  s   @r9   r   r   T  s    
"/ "/H)V!F;2.#`',/&-
)</(<V- .>@eN>;@(TcJJX`H &*]"] 
]~2h%N..` r;   r   c                     ^  \ rS rSr% SrSS0rS\S'     S   SU 4S jjjrS rS	 r	SS
 jr
S rS rS rS rS rSrU =r$ )r!  i	  z?
Object to convert one Part stream to a <part> tag on .parse()
previousPartStaffInGroupa=  
            If the part being exported is a :class:`~music21.stream.base.PartStaff`,
            this attribute will be used to store the immediately previous `PartStaff`
            in the :class:`~music21.layout.StaffGroup`, if any. (E.g. if this is
            the left hand, store a reference to the right hand.)zdict[str, str]	_DOC_ATTRc                (  > [         TU ]  5         Uc  [        R                  " 5       nXl        X l        [        S5      U l        Uc2  [        R                  " 5       U l        SS/U l	        / U l
        SU l        OfUR                  b  UR                  O[        R                  " 5       U l        UR                  U l	        UR                  U l
        UR                  U l        S U l        S U l        S U l        S U l        S U l        UR"                  U l        g )Npartr   T)r  r{   r"   r^   r   r   r  rd   r  r   r  rx   r  instrumentStreamfirstInstrumentObjectlastDivisionsr/  r  )ry   partObjr   r  s      r9   r{   PartExporter.__init__
  s     	 ?kkmG07v>GM}}D),c
D%#%D  $D $*#5#5#A !' 2 2%+]]_  )/(C(CD%#)#9#9D  & 3 3D?C%KO%)" " 37$22r;   c                   U R                   R                  SSS9  U R                  (       aq  U R                   R                  SS9S   U l         U R                   R	                  [         R
                  5      (       a  U R                  5         OJU R                  5         O9U R                   R	                  [         R
                  5      (       d  [        S5      eU R                  R                  5         U R                  5         U R                  R                  S[        U R                  R                   5      5        U R                   R	                  [         R
                  5       Hp  nU R#                  S[        UR$                  5      -   5        ['        XS9nU R                  Ul	         UR)                  5       nU R                  R3                  U5        Mr     U R                  $ ! [         ae  n[        UR$                  5      Ul        [-        U R                   [         R.                  5      (       a  U R                   R0                  Ul        UeS	nAff = f)
af  
Set up instruments, convert sounding pitch to written pitch,
create a partId (if no good one exists) and set it on
<part>, fixes up the notation
(:meth:`fixupNotationFlat` or :meth:`fixupNotationMeasured`),
setIdLocals() on spanner bundle. Run parse() on each measure's MeasureExporter and
append the output to the <part> object.

In other words, one-stop shopping.

:attr:`makeNotation` when False, will avoid running
:meth:`~music21.stream.base.Stream.makeNotation`
on the Part. Generally this attribute is set on `GeneralObjectExporter`
or `ScoreExporter` and read from there. Running with `makeNotation`
as False will raise `MusicXMLExportException` if no measures are present.
If `makeNotation` is False, the transposition to written pitch is still
performed and thus will be done in place.

>>> from music21.musicxml.m21ToXml import PartExporter
>>> noMeasures = stream.Part(note.Note())
>>> pex = PartExporter(noMeasures)
>>> pex.makeNotation = False
>>> pex.parse()
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
Cannot export with makeNotation=False if there are no measures
Tr  r   r   z>Cannot export with makeNotation=False if there are no measuresr6  zMeasure r  N)r"   r  rx   splitAtDurationsr   r`   fixupNotationMeasuredfixupNotationFlatr,   r  setIdLocalsinstrumentSetupr  r:  rR   r  partIdr%  r   MeasureExporterr   measureNumberr   r^   r   r   )ry   r   measureExporter	mxMeasurer  s        r9   r   PartExporter.parse2
  s   @ 	""44"H ++66t6DQGDK{{--fnn==**,&&(//??)PR R 	&&( 	s4#=#=#D#DEF//?A"":AHH#=>-a=O,0,>,>O)+113	 LL	* @ || + "%ahh-dkk6;;77!%!5!5AJs   'G""
I,A IIc                    U R                   b_  [        U R                  5      U R                   R                  ;   a2  U R                   R                  [        U R                  5         U l        OU R                  R                  SSS9U l        U R                  S   U l        U R                   b0  U R                   R                   Vs/ s H  oR                  PM     nnOU R                  R                  /nU R                  R                  nX2;   d  Uc  U R                  R                  5         U R                  5       nU(       a  g[        5       nU R                   GH[  n[        U5      U;   a  M  UR                  [        U5      5        UR                  b  UR                  U R                  ;   a   UR!                  U R                  S9  U R                  R-                  UR                  5        UR.                  b5  U R                   (       a4  UR.                  U R                   R0                  ;   a  UR3                  5         U R                   c  M  U R                   R0                  R-                  UR.                  5        X`R                  L d  GM6  U R                   R                  R-                  U5        GM^     gs  snf ! ["        R$                   a)  n[&        R)                  [+        U5      5         SnAGN.SnAff = f)a  
Sets self.instrumentStream and self.firstInstrumentObject for the stream,
checks for a unique midiChannel and then blocks it off from future use.

>>> p = converter.parse('tinyNotation: 4/4 c1 d1 e1')
>>> p.getElementsByClass(stream.Measure)[0].insert(0, instrument.Clarinet())
>>> p.getElementsByClass(stream.Measure)[1].insert(0, instrument.BassClarinet())
>>> PEX = musicxml.m21ToXml.PartExporter(p)
>>> PEX.instrumentStream is None
True
>>> PEX.firstInstrumentObject is None
True
>>> PEX.instrumentSetup()
>>> PEX.instrumentStream
<music21.stream.Part 0x10ae02780>

The "P" signifies that it is the main instrument associated with a Part.

>>> PEX.instrumentStream.show('text')
{0.0} <music21.instrument.Clarinet 'P...: Clarinet'>
{4.0} <music21.instrument.BassClarinet 'Bass clarinet'>
NTr  r   r   )usedChannels)r   r6  r"   r  r  getInstrumentsr  r  r	  partIdRandomize#mergeInstrumentStreamPartStaffAwarer:  r  addmidiChannelr  autoAssignMidiChannelr   InstrumentExceptionenvironLocalr   rR   r   instrumentIdr  instrumentIdRandomize)ry   r0  
instIdListfirstInstIdshould_short_circuitseen_instrument_classesthisInstrumentr  s           r9   r  PartExporter.instrumentSetupx
  sK   0 ;;"r$++$++:Y:Y'Y %)KK$C$CBt{{O$TD! %)KK$>$>T[_$>$`D!%)%:%:1%=";;",0KK,F,FG,Fq((,FJGJ++..)J0077$(;&&668#GGI"%%"33N N#'>>#''^(<=**2%11T5I5II."88dFZFZ8[   ''(B(BC ++3KK&33t{{7S7SS446 {{&,,33N4O4OP!%?%??KK..55nE; 4 H0 $77 . %%c!f--.s   J;1K  K=K88K=c                T   U R                   c  gU R                   R                  (       d  gU R                   R                   H  nUR                  U R                  5      (       a  USS  H  nUR	                  SSS9nU =R
                  U-  sl        U R
                  b  [        R                  " U R
                  SS9  U R                   (       d  Mg  U R
                  c  Mv  [        U5      nU R
                  U R                   R                  U'   M     M  U R                  U;   d  M    g   g)a!  
Merges instrument streams from subsequent parts in a PartStaff group.

Does nothing in the normal case of single staves.

Returns whether instrument processing should short circuit,
which is False for the general case and True for subsequent
PartStaff objects after the first in a group.
NFr   Tr  r   )
r   r  r  r"   r  r  r   deduplicater6  r  )ry   joined_groupsubsequent_staffother_instrumentsnext_ids        r9   r  0PartExporter.mergeInstrumentStreamPartStaffAware
  s    ;; {{'' KK44L##DKK00(4QR(8$(8(G(G&*D )H ):%))->>),,8"..t/D/DdS {{{t'<'<'H"$%5"6CGCXCX77@ )9 ,
 ' 5( r;   c                    U R                   nUR                  5         UR                  U R                  U R                  SS9  UR
                  U l        g)zB
Runs makeNotation on a flatStream, such as one lacking measures.
T)r  r   r   N)r"   makeMutablerx   r  r   r  )ry   r  s     r9   r  PartExporter.fixupNotationFlat
  sR     {{ 	d&6&6/3/H/H"& 	 	(& "//r;   c                2   U R                   nUR                  [         R                  5      nUR                  5       nU(       d  g[	        US5      (       aX  UR
                  cK  UR                  5         UR                  [
        R                  5      nU(       a  UR                  5       Ul        [	        US5      (       aX  UR                  cK  UR                  5         UR                  [        R                  5      nU(       a  UR                  5       Ul        [	        US5      (       aX  UR                  cK  UR                  5         UR                  [        R                  5      nU(       a  UR                  5       Ul        UR                  R                  5       (       d  UR!                  SS9  UR                  R"                  (       d   UR%                  SS9  UR                  R2                  (       d=  U H7  nU/UR4                  Q H!  n	[         R6                  R9                  U	SS9  M#     M9     U R:                  (       d  UR:                  U l        gg! [&        R(                   a.  n[*        R,                  " [.        [1        U5      5         SnANSnAff = f)a  
Checks to see if there are any attributes in the part stream and moves
them into the first measure if necessary.

Checks if makeAccidentals is run, and haveBeamsBeenMade is done, and
tuplets have been made.

* Changed in v7: no longer accepts `measureStream` argument.
Nr   keySignaturer   Tr   )r"   r   r`   r   rt  r   r*  r   r-  r   KeySignaturer   r   TimeSignaturestreamStatushaveAccidentalsBeenMademakeAccidentalsbeams	makeBeamsr   StreamExceptionr   r   r-   rR   tupletsvoicesrx   r   r  )
ry   r  measuresfirst_measure
outerClefsouterKeySignaturesouterTimeSignaturesser   m_or_vs
             r9   r  "PartExporter.fixupNotationMeasured
  s    {{**6>>: ( =&))m.@.@.H%%'00;J%/%5%5%7"=.11m6P6P6X%%'!%!8!89I9I!J!-?-E-E-G*=/22}7R7R7Z%%'"&"9"9%:M:M"N".A.G.G.I+   88::   .  &&8t,
   (( n188nF''::64:P -  !!!%!3!3D "  // 8 os2w778s   	I J($JJc                   U R                   n[        S5      nUR                  SU R                  R	                  S5      5        [        US5      n[        US5      (       a  UR                  b  UR                  Ul        O[        R                  Ul        UR                  (       a-  UR                  R                  (       d  UR                  SS5        [        U R                   S5      (       ah  UR                  b[  [        US5      nUR                  Ul        UR                  (       a-  UR                  R                  (       d  UR                  SS5        [        5       nU R                   Hw  n[!        U5      U;   a  M  UR"                  c  UR$                  c  UR&                  c  M=  UR)                  U R+                  U5      5        UR-                  [!        U5      5        My     [        5       nU R                   H|  n[!        U5      U;   a  M  UR&                  c!  [/        U[0        R2                  5      (       d  MB  UR)                  U R5                  U5      5        UR-                  [!        U5      5        M~     U$ )	z
make a <score-part> from a music21 Part object and a parsed mxPart (<part>) element.

contains details about instruments, etc.

called directly by the ScoreExporter as a late part of the processing.
z
score-partr6  z	part-namer   ra  rb  r   zpart-abbreviation)r"   r   r:  r  getr   rt  r   r&   r   r4  r$   printPartNamer   printPartAbbreviationr  r  instrumentNameinstrumentAbbreviationmidiProgramr   instrumentToXmlScoreInstrumentr  r   r   UnpitchedPercussioninstrumentToXmlMidiInstrument)ry   r  r  
mxPartNamemxPartAbbreviationr  insts          r9   r  PartExporter.getXmlScorePart@  s    {{l+dll..t45[9
4$$)B"mmJO&//JO##DJJ,D,DNN>40 4;; 2338M8M8Y!+K9L!M&*&;&;#''

0P0P"&&~t<
 #&%))DDz44##/22>''3""4#F#Ft#LM'++DJ7 * #&% ))DDz44+z$
@^@^/_/_""4#E#Ed#KL'++DJ7 * r;   c                   [        S5      nUR                  S[        UR                  5      5        [	        US5      n[        UR
                  5      Ul        UR                  b&  [	        US5      n[        UR                  5      Ul        U$ )a4  
Convert an :class:`~music21.instrument.Instrument` object to a
<score-instrument> element and return it.

>>> i = instrument.Clarinet()
>>> i.instrumentId = 'clarinet1'
>>> i.midiChannel = 4
>>> PEX = musicxml.m21ToXml.PartExporter()
>>> mxScoreInstrument = PEX.instrumentToXmlScoreInstrument(i)
>>> PEX.dump(mxScoreInstrument)
<score-instrument id="clarinet1">
  <instrument-name>Clarinet</instrument-name>
  <instrument-abbreviation>Cl</instrument-abbreviation>
</score-instrument>

>>> i.instrumentName = "Klarinette 1."
>>> i.instrumentAbbreviation = 'Kl.1'
>>> mxScoreInstrument = PEX.instrumentToXmlScoreInstrument(i)
>>> PEX.dump(mxScoreInstrument)
<score-instrument id="clarinet1">
  <instrument-name>Klarinette 1.</instrument-name>
  <instrument-abbreviation>Kl.1</instrument-abbreviation>
</score-instrument>
zscore-instrumentr6  zinstrument-namezinstrument-abbreviation)r   r:  rR   r  r   rD  r&   rE  )ry   r   mxScoreInstrumentmxInstrumentNamemxInstrumentAbbreviations        r9   rG  +PartExporter.instrumentToXmlScoreInstrument|  s    4 $$67dC$78%&79JK #A$4$4 5##/'12CE^'_$,/0H0H,I$)
 ! r;   c                b   [        S5      nUR                  S[        UR                  5      5        UR                  c   UR                  U R                  5        [        US5      n[        UR                  S-   5      Ul
        UR                  b)  [        US5      n[        UR                  S-   5      Ul
        [        U[        R                  5      (       a6  UR                  b)  [        US5      n[        UR                  S-   5      Ul
        U$ ! [        R                   a
    SUl         Nf = f)a6  
Convert an instrument object to a <midi-instrument> tag and return the element

>>> i = instrument.Clarinet()
>>> i.instrumentId = 'clarinet1'
>>> i.midiChannel = 4
>>> PEX = musicxml.m21ToXml.PartExporter()
>>> mxMidiInstrument = PEX.instrumentToXmlMidiInstrument(i)
>>> PEX.dump(mxMidiInstrument)
<midi-instrument id="clarinet1">
  <midi-channel>5</midi-channel>
  <midi-program>72</midi-program>
</midi-instrument>

>>> m = instrument.Maracas()
>>> m.instrumentId = 'my maracas'
>>> m.midiChannel  # 0-indexed
9
>>> m.percMapPitch
70
>>> PEX = musicxml.m21ToXml.PartExporter()
>>> mxMidiInstrument = PEX.instrumentToXmlMidiInstrument(m)
>>> PEX.dump(mxMidiInstrument)  # 1-indexed in MusicXML
<midi-instrument id="my maracas">
  <midi-channel>10</midi-channel>
  <midi-unpitched>71</midi-unpitched>
</midi-instrument>
zmidi-instrumentr6  r   zmidi-channelr   zmidi-programzmidi-unpitched)r   r:  rR   r  r  r  r  r   r  r   r&   rF  r   r   rH  percMapPitch)ry   r   mxMidiInstrumentmxMidiChannelmxMidiProgrammxMidiUnpitcheds         r9   rI  *PartExporter.instrumentToXmlMidiInstrument  s   < ##45T3q~~#67== "''(<(<= ##3^D !23 ==$&'7HM!$Q]]Q%6!7Ma7788Q^^=W()9;KLO#&q~~'9#:O   !  33 " !"s   D D.-D.)r  r  r   rx   r  r  r   r  r   r  r/  r"   r  NN)r  z!stream.Part | stream.Score | Noner   zScoreExporter | Noner   r  )r   r   r   r   r   r  __annotations__r{   r   r  r  r  r  r  rG  rI  r   r  r  s   @r9   r!  r!  	  sz     	# %D!I~  ;?.2%37%3+%3 %3NDLLF\%N0:44l:x%!N3  3 r;   r!  c                  R  ^  \ rS rSr\" / SQ5      r\" / SQ5      rSS1r  SW   SXU 4S jjjrS r	S r
SYS	 jrSYS
 jrSS.     SZS jjr\R                  4 S[S jjrS\S jr    S]S jr\S^S j5       r\S^S j5       rS_S jr  S`       SaS jjrSbS jr          ScS jrS`SdS jjr      SeS jrSfS jrSgS jrShS jrSiS jr  S`       SjS jjr SkS jr!SlS jr"SmS  jr#SkS! jr$ Sb       SnS" jjr%SoS# jr&S$ r'SpS% jr(  S`       SqS& jjr)SrS' jr*SrS( jr+SsS) jr,StS* jr-    SuS+ jr.S, r/SvS- jr0\SwS. j5       r1\SxS/ j5       r2SyS0 jr3SzS1 jr4S2S3.     S{S4 jjr5 SbS2S5.       S|S6 jjjr6 SbS2S5.       S}S7 jjjr7S~S8 jr8SS9 jr9SS: jr:SS; jr;SS< jr<SS= jr=SS> jr>    SS? jr?      SS@ jr@  S       SSA jjrASSB jrBSC rCSD rDSE rESF rFSG rGSH rHSI rISJ rJSK rKSL rLSSM jrMSN rNSO rOSbSP jrPSQ rQSbSR jrRSS rSST rTSU rUSVrVU =rW$ )r
  i  ))r   	noteToXml)NoChordnoChordToXml)ChordWithFretBoardchordWithFretBoardToXml)ChordSymbolchordSymbolToXml)RomanNumeralromanNumeralToXml)	ChordBase
chordToXml)	UnpitchedunpitchedToXml)r  	restToXml)rl   dynamicToXml)Segno
segnoToXml)Coda	codaToXml)	TempoTexttempoIndicationToXml)MetronomeMarkrr  )MetricModulationrr  )TextExpressiontextExpressionToXml)RepeatExpressionrv  )RehearsalMarkrehearsalMarkToXml)PedalObjectpedalObjectToXml))r   	clefToXml)r.  keySignatureToXml)r/  timeSignatureToXml
LayoutBaseBarlinec                  > [         TU ]  5         Uc  [        R                  " 5       U l        OXl        X l        [        S5      U l        [        R                  U l	        [        U R                  U R                  5        S U l        S U l        SU l        SU l        S U l        SU l        SU l        0 U l        / U l        Uc  [(        R*                  " 5       U l        OUR,                  U l        U R,                  U l        g )Nr   r   r   )r  r{   r"   r`   r   r   r  r   divisionsPerQuartercurrentDivisionssynchronizeIdstranspositionIntervalmxTransposemeasureOffsetStartoffsetInMeasurecurrentVoiceIdnextFreeVoiceNumbernextArpeggioNumberarpeggioNumbers
rbSpannersr!   SpannerBundler  objectSpannerBundle)ry   
measureObjr   r  s      r9   r{   MeasureExporter.__init__  s     	 ..*DK$Ky) ( < < 	t||T[[1 &*""%",0() '(KM79>!(!6!6!8D!'!5!5D#'#5#5 r;   c                   U R                  5         U R                  5         U R                  5         U R                  5         U R	                  5         U R                  5         U R                  5         U R                  5         U R                  $ )z
main parse call.

deals with transposing, repeat brackets, setting measureNumber and width,
the first mxPrint, the first <attributes> tag, the left barline, parsing all internal
elements, setting the right barline, then returns the root <measure> tag.
)	setTransposesetRbSpannerssetMxAttributes
setMxPrint&setMxAttributesObjectForStartOfMeasuresetLeftBarlinemainElementsParsesetRightBarliner  r  s    r9   r   MeasureExporter.parse  sl     	335 ||r;   c                   U R                   nUR                  5       (       d  U R                  USS9  gUR                  [         R                  5      R                  5       nU R                  USS9  [        UR                  5      n[        U5       H+  u  pEU[        U5      S-
  :X  a  SnOSnU R                  XVS9  M-     g)zP
deals with parsing all the elements in a stream, whether it has voices or not.
FbackupAfterwardsNTr   )	r"   	hasVoicesparseFlatElementsgetElementsNotOfClassrb   r  r7  rC  r  )ry   r   nonVoiceMeasureItems	allVoicesr   r   r  s          r9   r  !MeasureExporter.mainElementsParse2  s     KK {{}}""1u"= 66v||DKKM3dKN	i(DAC	NQ&&#( #'  ""1"H )r;   c                   [        [        XR                  -  5      5      nU(       aX  [        S5      n[	        US5      n[        U5      Ul        U R                  R                  U5        U =R                  U-  sl	        gg)ab  
Moves self.offsetInMeasure forward by an OffsetQL, appending the appropriate
<forward> tag (expressed in divisions) to self.xmlRoot.

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> len(MEX.xmlRoot)
0
>>> MEX.moveForward(1)
>>> MEX.dump(MEX.xmlRoot)
<measure>
  <forward>
    <duration>10080</duration>
  </forward>
</measure>
>>> len(MEX.xmlRoot)
1

forwardr   N
r7  roundr  r   r   rR   r&   r  r   r  )ry   byOffsetamountToMoveForward	mxForward
mxDurations        r9   moveForwardMeasureExporter.moveForwardK  sk    & $'uX8M8M-M'N#O	*I#Iz:J!"56JOLL	*  H,  r;   c                   [        [        XR                  -  5      5      nU(       aX  [        S5      n[	        US5      n[        U5      Ul        U R                  R                  U5        U =R                  U-  sl	        gg)a`  
Moves self.offsetInMeasure backward by an OffsetQL, appending the appropriate
<backup> tag (expressed in divisions) to self.xmlRoot.

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> len(MEX.xmlRoot)
0
>>> MEX.moveBackward(2)
>>> MEX.dump(MEX.xmlRoot)
<measure>
  <backup>
    <duration>20160</duration>
  </backup>
</measure>
>>> len(MEX.xmlRoot)
1
backupr   Nr  )ry   r  amountToBackupmxBackupr  s        r9   moveBackwardMeasureExporter.moveBackwardf  si    $ "%3H3H(H"IJx(H#Hj9J!.1JOLL)  H,  r;   Fr  c               B  ^  ST l         [        U[        R                  5      (       a  [        UR                  [
        5      (       a5  UR                  [        R                  :  a  UR                  nUS-   T l        OP[        UR                  [
        5      (       a"  T R                  nT =R                  S-  sl        OUR                  nOSnUT l	        [        UR                  [        R                  5      5      n[        U5      nU GH  nUR                  US   5      nUT R                   -
  nUS:  a(  [!        S U 5       5      (       a  T R#                  U5        / n	U H  n
[        U
[        R                  5      (       a  M$  [        U
[$        R&                  5      (       aL  [        U
[(        R*                  5      =n(       a  U(       a$  U
R,                  (       a  U	R/                  U
5        M  U(       a"  T R1                  U
[2        R4                  5        M  T R1                  U
[2        R6                  5        M     U	 H  nUR8                  (       a7  UR:                  R<                  (       a  UR>                  R@                  S:X  a  MK  U(       a"  T R1                  U[2        R4                  5        Mt  T R1                  U[2        R6                  5        M     GM     T R                   nU(       a  T R                   (       a  T RC                  T R                   5        [        U5      nU H  n[!        U 4S jU 5       5      (       d  M  UR                  US   5      nUT R                   -
  nUS:  a  T R#                  U5        U H#  n
T R1                  U
[2        RD                  5        M%     M     U(       a-  T R                   (       a  T RC                  T R                   5        O7T R                   U:  a'  T R#                  [G        UT R                   -
  5      5        ST l	        g)a  
Deals with parsing all the elements in .elements, assuming that .elements is flat.

`m` here can be a Measure or Voice, but it must be flat.

If m is a 'Voice' class, we use the .id element to set self.currentVoiceId and then
send a backup tag to go back to the beginning of the measure.

Note that if the .id is high enough to be an id(x) memory location, then a small
voice number is used instead.
r   r   Nr   c              3  v   #    U  H/  n[        U[        R                  [        R                  45      v   M1     g 7fr   )r   r   rf   r   r   )r/  rz   s     r9   r1  4MeasureExporter.parseFlatElements.<locals>.<genexpr>  s-      /XNVsJsT%5%5tyy$ABBhs   79r5   c              3  F   >#    U  H  nTR                  U5      v   M     g 7fr   )_hasRelatedSpanners)r/  rz   ry   s     r9   r1  r    s     MHS433C88Hs   !)$r  r   r"   rb   r6  r7  r   r8  r  r  r  r   r!   SpannerAnchorr#   elementOffsetanyr  r   rf   r   HarmonywriteAsChordr   parseOneElementr   NONENORMALisRestr$   rc  r   r  r  RELATED_ONLYr   )ry   r   r  voiceIdhasSpannerAnchorsobjIteratorobjGroupgroupOffsetoffsetToMoveForwardnotesForLaterrz   isHarmr   firstPassEndOffsetInMeasures   `             r9   r  !MeasureExporter.parseFlatElements  s5   "  #a&&!$$$$0\0\)\$$+2Q;(ADD#&& 22((A-($$G% #'q';';G<Q<Q'R"S ;I:K#H//(1+6K"-0D0D"D #Q&3 /XNV/X ,X ,X   !45M c7#8#899 c4#3#344#-c7??#CCC3#3#3!((-(,,S.2E2EF,,S.2G2GH#  & #88 9 9ajjooQ`>` $((N,?,?@((N,A,AB #? $` 150D0D###!!$"6"67(+K'MHMMMoohqk:&1D4H4H&H#&*$$%89 $C((n.I.IJ $ ( ##!!$"6"67 ##&AA  (CdFZFZ(Z![\"r;   c                   U R                   nU R                  R                  U5      U l        U R	                  U5      u  pEU[
        R                  :w  a  U H  nUR                  U5        M     U[
        R                  :w  Ga  UR                  nSU;   a;  [        USS5      (       a)  U =R                  UR                  R                  -  sl        UR                  R                  S:X  a6  UR                  R                  UR                  l        UR                  5       nOP[!        UR                  R"                  5      S:  a*  UR                  R%                  SS9  UR                  5       nOU/nSn	U R&                  R)                  5        H*  u  pX;   d  M  [        X5      nU H  nU" U5        M     Sn	  O   U R*                  R)                  5        H3  u  pX;   d  M  [        X5      nU H  nU R-                  X5        M     Sn	  O   U	SL a8  U H  n
XR.                  ;   d  M  Sn	  O   U	SL a  [0        R3                  SU/5        O^U(       aW  UR                  S	:  aG  SUR                  ;   a7  [        USS5      (       a%  U R5                  UR                  R                  5        U[
        R                  :w  a  U H  nUR                  U5        M     g
g
)zT
parse one element completely and add it to xmlRoot, updating
offsetInMeasure, etc.
rf   r  Tr5   r   r   Fzdid not convert objectr   N)r  r  getBySpannedElementr  relatedSpannersr   r  r   r  r   rP   r  r   r   r  r  r  	dotGroupssplitDotGroupsclassesToMethodsr   wrapAttributeMethodClasseswrapObjectInAttributesignoreOnParseClassesr  
printDebugr  )ry   rz   appendSpannersrootpreListpostListr#  r   objListparsedObject	classNamer   r   os                 r9   r  MeasureExporter.parseOneElement  su    ||#'#5#5#I#I##N  005 ^000B  ^888kkG'GC,N,N$$(B(BB$ ||  O3-0\\-G-G*..0S\\++,q0++D+9..0 % L'+'<'<'B'B'D#	'"42D$Q %#'L (E (,'F'F'L'L'N#	'"42D$33A< %#'L (O u$!(I $=$=='+ ")  5( ++-Es,KL C--1 !CKK/GCQU4V4V$$S\\%?%?@ ^000B  1r;   c                N   U R                   R                  U5      nU(       d  gUR                  [        R                  [
        R                  [        R                  [        R                  45       H1  nUR                  U5      (       d  UR                  U5      (       d  M1    g   g)z
returns True if and only if:
(1) there are spanners related to the object that should appear before the object
to the <measure> tag, or (2) there are spanners related to the object that should
appear after the object in the measure tag.
FT)r  r  r  r!   Ottavar   DynamicWedgeLiner   	PedalMarkr  r  )ry   rz   r  thisSpanners       r9   r  #MeasureExporter._hasRelatedSpannersR  s     **>>sC )33W^^5=5J5J5<\\5@5J5J5L MK ""3'';+=+=c+B+BM r;   c                *   S nU R                   nU(       d  g/ n/ nSSSSS.nUR                  5        GHZ  u  pxUu  pUR                  U5       GH:  nU" X5       GH)  n[        U	5      n[	        X5        UR                  S[        UR                  5      5        US	:X  a%  UR                  S
[        UR                  5      5        US:X  a  U R                  X{5      nOUS:X  a  U R                  X{5      nO0 nSU;   a  UR                  S[        US   5      5        U
 H.  nX;   d  M
  X   c  M  UR                  U[        X   5      5        M0     [        S5      n[	        UU5        [        US5      (       a2  UR                  b%  UR                  S[        UR                  5      5        [        US5      nUR                  U5        US:X  a  UR                  U5        OUR                  U5        US:X  d  GM  US:X  d  GM  [        R                   (       a!  [#        U[$        R&                  5      (       d   eUR(                  [$        R*                  R,                  :X  d  GM  U R/                  U5      nUR                  U5        GM,     GM=     GM]     XE4$ )z
return two lists or empty tuples:
(1) spanners related to the object that should appear before the object
in the <measure> tag. (2) spanners related to the object that should appear after the
object in the <measure> tag.
c                    [        U 5      S:X  a  SnU$ U R                  U5      (       a  U R                  U5      (       a  SnU$ U R                  U5      (       a  SnU$ U R                  U5      (       a  SnU$ SnU$ )Nr   )r   last)r   )r  r   )r  r  r  )r  innerTargetprocs      r9   getProc0MeasureExporter.relatedSpanners.<locals>.getProcq  s    2w!|( K ::k**ryy/E/E,D K ZZ,,%D
 K	 YY{++$D K DKr;   )r   r   )zoctave-shift)size)wedge)spread)r  )line-end
end-length)pedal)linesignabbreviated)r  r  r  r  r   r  	line-typer   r  r  r  	placementdirection-typer  )r  r   r  r   r  r:  rR   idLocallineType_spannerStartParameters_spannerEndParametersrt  r  r   r   r  r  r   r   r  	pedalForm	PedalForm
SymbolLinemakePedalResumeLineXml)ry   targetr  r  r  r  	paramsSetm21spannerClass	infoTuplemxTagparameterSetr  posSub	mxElementspannerParamsattrNamemxDirectionmxDirectionTypemxPedalLines                      r9   r  MeasureExporter.relatedSpannersg  sG   	 00!#"$
  ;%;D"L	 +4//*;&O"+E,77H%k:F 'I"9:MM(C0C0C,DE&&0!k3{7K7K3LM((,(D(D_(b6)(,(B(B?(`(*.!fc-2G.HI$0#49P9\%MM(C8O4PQ %1 #*+"6K";< {K88[=R=R=^#S9N9N5OP&0>N&OO#**95 ({3 4&+5&G:K??#-k;;P;P#Q#QQ#Q&00K4I4I4T4TT $ ; ;K H ( $NN;7] ;  I +<f   r;   c                   SS0nU S:X  a]  [         R                  (       a!  [        U[        R                  5      (       d   eUR                  5       US'   UR                  SS9US'   U$ U S:X  aV  [         R                  (       a!  [        U[        R                  5      (       d   eUR                  US'   UR                  US	'   U$ U S
:X  a  [         R                  (       a!  [        U[        R                  5      (       d   eUR                  US'   UR                  S:X  a  SUS'   UR                  (       a  SUS'   U$ UR                  US'    U$ U S:X  a  [         R                  (       a!  [        U[        R                   5      (       d   eUR"                  [        R$                  R&                  :X  a  SUS'   OSUS'   UR(                  [        R*                  R                  :X  a  SUS'   OSUS'   UR,                  (       a  SUS'   U$ )a  
Return a dict of the parameters for the start of this spanner required by MusicXML output.

>>> ssp = musicxml.m21ToXml.MeasureExporter._spannerStartParameters

>>> ottava = spanner.Ottava(type='8va')
>>> st = ssp('Ottava', ottava)

Note that an ottava up in MusicXML is type "down", perhaps the opposite of what
you might expect.  It means that the music should be engraved "down" from
where the sounding pitches would normally be engraved.

>>> st['type']
'down'
>>> st['size']
8

>>> cresc = dynamics.Crescendo()
>>> st = ssp('DynamicWedge', cresc)
>>> st['type']
'crescendo'
>>> st['spread']
0

>>> diminuendo = dynamics.Diminuendo()
>>> st = ssp('DynamicWedge', diminuendo)
>>> st['type']
'diminuendo'
>>> st['spread']
15
r  r  r  r  T)reverser  r  r  r  	crescendor   r  r  nienter  	sostenutor  r  r  )r  r  r   r!   r  shiftMagnitudeshiftDirectionr  	startTickstartHeightr   r  r  r  r  r   r  	pedalType	PedalType	Sostenutor  r   r  spannerClassr#  posts      r9   r  'MeasureExporter._spannerStartParameters  s   B #)'!28#!"gnn5555,,.DL,,T,:DLD C V#!"gll3333!||D!#D: 9 ^+!"h&;&;<<<<77DLww+%!"X99%*DN* ' "$X& % [(!"k&;&;<<<<||{44>>>*V  'V||{44999$V  %V~~&+]#r;   c                p   SS0nU S:X  aK  [         R                  (       a!  [        U[        R                  5      (       d   eUR                  5       US'   U$ U S:X  aV  [         R                  (       a!  [        U[        R                  5      (       d   eUR                  US'   UR                  US'   U$ U S:X  at  [         R                  (       a!  [        U[        R                  5      (       d   eUR                  S	:X  a  UR                  US
'   U$ SUS
'   UR                  (       a  SUS'   U$ U S:X  a  [         R                  (       a!  [        U[        R                  5      (       d   eUR                   [        R"                  R                  [        R"                  R$                  4;   a  SUS'   U$ SUS'   U$ )z
Return a dict of the parameters for the end of this spanner required by MusicXML output.

>>> ottava = spanner.Ottava(type='8va')
>>> en = musicxml.m21ToXml.MeasureExporter._spannerEndParameters('Ottava', ottava)
>>> en['type']
'stop'
>>> en['size']
8
r  r  r  r  r  r  r  r  r  r  r   r  r  r  r  r  )r  r  r   r!   r  r  r  endTick	endHeightr   r  r  r  r  r   r  r  r   r  r  s      r9   r  %MeasureExporter._spannerEndParameters  sw    #)&!18#!"gnn5555,,.DL0 / V#!"gll3333!zzD!#D& % ^+!"h&;&;<<<<ww+%!#X  "#X99%*DN  [(!"k&;&;<<<<|| 5 5 : :K<Q<Q<\<\]]$V
   %Vr;   c                    U R                   R                  US5      nUS:X  aF  U R                  nU =R                  S-  sl        U R                  S:  a  SU l        X R                   U'   U$ )Nr      )r  rA  r  )ry   arparpeggioNumbers      r9   getArpeggioNumber!MeasureExporter.getArpeggioNumber;  sh    "2266sB?R!44N##q(#&&+*+'(6  %r;   c                	   / nUb  UnOU R                   nU(       d  U$ U R                  XXE5        US:H  nU(       d  U$ / nUR                  S5       GH   n[        S5      n	UR	                  U5      (       az  U	R                  SS5        U R                  X5        U R                  X5        U R                  U	US5        UR                  b%  U	R                  S[        UR                  5      5        O+UR                  U5      (       a  U	R                  SS5        OM  U	R                  S	[        UR                  5      5        UR                  U	5        GM     UR                  S
5       GH  nUR                  S:X  a  Sn
OSn
[        U
5      nUR                  S	[        UR                  5      5        UR                  b%  UR                  S[        UR                  5      5        UR	                  U5      (       a:  UR                   b  [        UR                   5      Ul        UR                  SS5        O+UR                  U5      (       a  UR                  SS5        OM  [%        X5        UR                  U5        GM     UR                  S5       H  n[        S5      n[        UR&                  5      Ul        UR	                  U5      (       aE  UR                  SS5        UR                  b%  UR                  S[        UR                  5      5        O@UR                  U5      (       a  UR                  SS5        O[(        R+                  SX/5        UR                  U5        M     UR                  S5       GH/  n[        S5      nUR                  S	[        UR                  5      5        SnSnUR	                  U5      (       a=  UR                  SS5        SnUR                  b  UR                  SUR                  5        UR                  U5      (       a  USL a  SnOUR                  SS5        SnUSL a  M  UR                  U5        USL d  M  [        S5      nUR                  S	[        UR                  5      5        UR                  SS5        UR                  U5        GM2     U(       a6  [        S5      nU H  nUR                  U5        M     UR                  U5        U$ )a  
return a list of <notations> from spanners related to the object that should appear
in the notations tag (slurs, slides, etc.)

>>> n0 = note.Note('C')
>>> n1 = note.Note('D')
>>> trem = expressions.TremoloSpanner([n0, n1])
>>> m = stream.Measure()
>>> m.insert(0, trem)
>>> m.append(n0)
>>> m.append(n1)
>>> mex = musicxml.m21ToXml.MeasureExporter(m)
>>> out = mex.objectAttachedSpannersToNotations(n0, objectSpannerBundle=m.spannerBundle)
>>> out
[<Element 'ornaments' at 0x1114d9408>]
>>> mex.dump(out[0])
<ornaments>
  <tremolo type="start">3</tremolo>
</ornaments>

>>> out = mex.objectAttachedSpannersToNotations(n1, objectSpannerBundle=m.spannerBundle)
>>> mex.dump(out[0])
<ornaments>
  <tremolo type="stop">3</tremolo>
</ornaments>
r   Slurslurr  r  )zbezier-offsetzbezier-offset2zbezier-xzbezier-yz	bezier-x2z	bezier-y2r  r  r   	Glissando
continuousslide	glissandor  TremoloSpannertremoloz:spanner w/ a component that is neither a start nor an end.TrillExtensionz	wavy-lineFT	ornaments)r  %appendArpeggioMarkSpannersToNotationsr  r   r  r:  setLineStyler\  rC  r  rR   r  r  r   	slideTyper  labelr&   r  numberOfMarksr  r  )ry   rz   noteIndexInChordr  	notationssbisSingleNoteOrFirstInChordr6  r  mxSlurr  mxGlissandomxTrem
mxWavyLineisFirstOrLastisFirstANDLast
mxOrnGroupmxOrns                     r9   !objectAttachedSpannersToNotations1MeasureExporter.objectAttachedSpannersToNotationsF  s   @ $&	*$B))B223)X&6!&;" *	--'BV_Fzz#

67+!!&-  ,''(*)C* <<+JJ{C,=>3

66*JJxRZZ1V$' (* --,B|||+#!%.KOOHc"**o6{{&S-=>zz#88''*288}K$03/;+[)- -4 -- 01BY'Fb../FKzz#

67+<<+JJ{C,=>3

66* ''QSU[] V$ 2  -- 01B -JNN8S_5!M"Nzz#vw/ $<<+NN;= yy~~ D(%)NNN662$(M %Z(%$[1
xRZZ9vv.  ,= 2@  -J"!!%( #Z(r;   c                   / nUb  UnOU R                   nU(       d  U$ UR                  [        R                  5       H  n[	        S5      nUR                  U5      (       a  UR                  SS5        SUl        O+UR                  U5      (       a  UR                  SS5        OMi  UR                  SS5        UR                  U5        M     UR                  [        R                  5       H  n[	        S5      nUR                  U5      (       a  UR                  SS5        S	Ul        O+UR                  U5      (       a  UR                  SS5        OMi  UR                  SS5        UR                  U5        M     U$ )
a  
return a list of <technical> from spanners related to the object that should appear
in the technical tag of the notations tag (hammer-on, pull-off etc.)

>>> n0 = note.Note('C')
>>> n1 = note.Note('D')
>>> n2 = note.Note('F')
>>> n3 = note.Note('E')
>>> hammerOn01 = articulations.HammerOn([n0, n1])
>>> pullOff23 = articulations.PullOff([n2, n3])
>>> m = stream.Measure()
>>> m.insert(0, hammerOn01)
>>> m.insert(0, pullOff23)
>>> m.append(n0)
>>> m.append(n1)
>>> m.append(n2)
>>> m.append(n3)
>>> mex = musicxml.m21ToXml.MeasureExporter(m)
>>> out = mex.objectAttachedSpannersToTechnicals(n0, m.spannerBundle)
>>> out
[<Element 'hammer-on' at 0x102857f40>]
>>> mex.dump(out[0])
<hammer-on number="1" type="start">H</hammer-on>
>>> out = mex.objectAttachedSpannersToTechnicals(n3, m.spannerBundle)
>>> mex.dump(out[0])
<pull-off number="1" type="stop" />

OMIT_FROM_DOCS

The other permutations of class and first/last:

>>> out = mex.objectAttachedSpannersToTechnicals(n1, m.spannerBundle)
>>> mex.dump(out[0])
<hammer-on number="1" type="stop" />
>>> out = mex.objectAttachedSpannersToTechnicals(n2, m.spannerBundle)
>>> mex.dump(out[0])
<pull-off number="1" type="start">P</pull-off>
z	hammer-onr  r  Hr  r   rQ  zpull-offP)r  r  r   HammerOnr   r  r:  r&   r  r   PullOff)ry   rz   r  
technicalsr>  r  
mxHammerOn	mxPullOffs           r9   "objectAttachedSpannersToTechnicals2MeasureExporter.objectAttachedSpannersToTechnicals  s1   N 
*$B))B-- 6 67B -Jzz#vw/"%
3vv.NN8S)j) 8 -- 5 56B
+Izz#fg.!$	3ff-MM(C(i( 7 r;   c                   UR                  [        R                  5       GHV  nUR                  U5      (       d  M  Un[	        U[
        R                  5      (       a  X   nS nUR                  S:X  aV  UR                  5       u  pXhL a  [        S5      nUR                  SS5        OYXiL a  [        S5      nUR                  SS5        O7[        S5      nUR                  S:w  a  UR                  SUR                  5        Ub\  [        U5      S	:  d!  [        U5      S	:X  a>  [        US
   5      S	:  a,  U R                  U5      n
UR                  S[        U
5      5        Uc  GME  UR                  U5        GMY     g )Nnon-arpeggionon-arpeggiater  r  r  
arpeggiatenormalr  r   r   r   )r  r   ArpeggioMarkSpannerhasSpannedElementr   r   Chordr  noteExtremesr   r:  r  r*  rR   r   )ry   rz   r<  r=  r>  amssub_obj
mxArpeggiomin_notemax_noter)  s              r9   r7  5MeasureExporter.appendArpeggioMarkSpannersToNotations*  sA    ==!@!@AC((-- G#u{{++/'+Jxx>)%(%5%5%7"&!()9!:JNN684(!()9!:JNN651$\2
88x'NN;9%3s8a<CHMcRUVWRXk\]o
 '+&<&<S&Ax^)<=%  ,A Br;   c                F   US:g  n[         n[        S5      nUc  UnOUnU R                  Xa5        U R                  Xg5        [        U[        R
                  5      (       a\  UR                  5       (       aG  UR                  R                  b0  UR                  R                  S-  S-  nUR                  SUS 5        [        Xa5        UR                  n	[        U	[        R                  5      (       a{  [        US5      n
 U	R                  S	;   a  U" XS
[        R                   S9  U	R"                  b  U" U	U
S[        R$                  S9  U	R&                  b  U" U	U
S[        R$                  S9  U R/                  Xa5        UR0                  (       a+  UR2                  R4                  SL a  UR                  SS5        UR6                   H'  nSUR8                  ;   d  M  UR                  SS5        M)     U(       a  [        US5        [;        US5      (       aR  [<        R>                  " [        R@                  U5      nU RC                  URD                  5      nURG                  U5        OURH                  (       a  [        US5        U	RJ                  SLa"  U RM                  U	5      nURG                  U5        URN                  b5  U RQ                  URN                  5      nU H  nURG                  U5        M     [        U[        R
                  5      (       a  U RS                  XU5        U RU                  Xa5        U RV                  b&  [        US5      n[Y        U RV                  5      Ul-        [        S5      nU	RJ                  SL a  U	R\                  S:X  a  SUl-        O [_        U	R\                  5      Ul-        U Ri                  UUSS5        URG                  U5        [k        U	Rl                  5       H  n[        US5        M     [        U[        R@                  5      (       aq  URD                  Rn                  bZ  URD                  Rn                  Rp                  S;   a6  U Rs                  URD                  Rn                  5      nURG                  U5        [u        U	Rv                  5      S:X  a0  U Ry                  U	Rv                  S   5      nURG                  U5        O[u        U	Rv                  5      S:  a  [z        R|                  " U	R                  5       5      R                  S 5      n[        R                  " UR                  UR                  5      nU Ry                  U5      nURG                  U5        SnUS!L aa  [        U[        R
                  5      (       aB  UR                  S":w  a2  [<        R>                  " [        R
                  U5      nUR                  nOdXqLa`  [        U[        R
                  5      (       aA  UR                  S":w  a1  [<        R>                  " [        R
                  U5      nUR                  nSnUR0                  (       aV  [        UR2                  [2        R                  5      (       a-  UR2                  R                  b  UR2                  R                  nUc  U(       a  [        US#5      nUcE  UR                  [        R                  5      =n(       a  UR                  UR                  5      nOS$nOUnUS%:X  a  S&nUUl-        U(       a$  U R                  UU5        U R                  UU5        U R                  XaU5        US!L aa  [        U[        R
                  5      (       aB  UR                  b5  U R                  UR                  5      nU H  nURG                  U5        M     U R                  XU5      nUS!L aE  [        U	Rv                  5       H,  u  nn U R                  U US-   5      n!UR                  U!5        M.     U(       a&  [        US'5      n"U H  n#U"RG                  U#5        M     US!L aD  UR                   H4  n$U$RZ                  c  M  U R                  U$5      n%URG                  U%5        M6     U R                  RG                  U5        U$ ! [(         a    [*        R-                  SU	 35         GNff = f! [`         a?    [        U[        Rb                  5      (       a  [d        Rf                  " U5      (       a   GNee f = f)(a  
Translate a music21 :class:`~music21.note.Note` or a Rest into a
ElementTree, note element.

Note that, some note-attached spanners, such
as octave shifts, produce direction (and direction types)
in this method.

>>> n = note.Note('D#5')
>>> n.quarterLength = 3
>>> n.volume.velocityScalar = 0.5
>>> n.style.color = 'silver'

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> len(MEX.xmlRoot)
0
>>> mxNote = MEX.noteToXml(n)
>>> mxNote
<Element 'note' at 0x10113cb38>
>>> MEX.dump(mxNote)
<note color="#C0C0C0" dynamics="70.56">
  <pitch>
    <step>D</step>
    <alter>1</alter>
    <octave>5</octave>
  </pitch>
  <duration>30240</duration>
  <type>half</type>
  <dot />
  <accidental>sharp</accidental>
  <notehead color="#C0C0C0" parentheses="no">normal</notehead>
</note>
>>> len(MEX.xmlRoot)
1

>>> r = note.Rest()
>>> r.quarterLength = 1/3
>>> r.duration.tuplets[0].type = 'start'
>>> mxRest = MEX.noteToXml(r)
>>> MEX.dump(mxRest)
<note>
  <rest />
  <duration>3360</duration>
  <type>eighth</type>
  <time-modification>
    <actual-notes>3</actual-notes>
    <normal-notes>2</normal-notes>
    <normal-type>eighth</normal-type>
  </time-modification>
  <notations>
    <tuplet bracket="yes" number="1" placement="above" type="start">
      <tuplet-actual>
        <tuplet-number>3</tuplet-number>
        <tuplet-type>eighth</tuplet-type>
      </tuplet-actual>
      <tuplet-normal>
        <tuplet-number>2</tuplet-number>
        <tuplet-type>eighth</tuplet-type>
      </tuplet-normal>
    </tuplet>
  </notations>
</note>
>>> len(MEX.xmlRoot)
2

>>> n.notehead = 'diamond'
>>> n.articulations.append(articulations.Pizzicato())
>>> mxNote = MEX.noteToXml(n)
>>> MEX.dump(mxNote)
<note color="#C0C0C0" dynamics="70.56" pizzicato="yes">
  ...
  <notehead color="#C0C0C0" parentheses="no">diamond</notehead>
</note>

Notes with complex durations need to be simplified before coming here
otherwise they raise :class:`MusicXMLExportException`:

>>> nComplex = note.Note()
>>> nComplex.duration.quarterLength = 5.0
>>> mxComplex = MEX.noteToXml(nComplex)
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
Cannot convert complex durations to MusicXML.
Try exporting with makeNotation=True or manually running splitAtDurations()

TODO: Test with spanners
r   r   Nd   g?>?r   z.2fgrace)TFslashr  zsteal-time-previouszsteal-time-followingz6Duration set as Grace while not being a GraceDuration Tzprint-spacingr  	Pizzicato	pizzicator   r   restvoicer  r7   eighthr  noteSizedot)TNr   i  FunspecifiedstemupnoStemnoner=  )Xr  r   rX  r   r   NotResthasVolumeInformationvolumevelocityScalarr:  r  r   GraceDurationr   rf  r*   r8  stealTimePreviousfractionToPercentstealTimeFollowingrQ   r  r   rd  r4  r$   rc  r   r   rt  r  castr   
pitchToXmlr   r   r  isGracedurationXmlr'   tieToXmlTiesetNoteInstrumentr  r  rR   r&   r  r:   r,   r  r(   isFullMeasureRestrC  r   dotsr  displayStatusr  r  r6  tupletToTimeModification	fractionsFractionaggregateTupletMultiplierlimit_denominatorTupletdenominator	numeratorstemDirection	NoteStyle	stemStyler   r   r   getStemDirectionForPitchespitchesr^  r\  dealWithNoteheadr3  
beamsToXmlnoteToNotationsrC  tupletToXmlTupletextendlyrics
lyricToXmlr  )&ry   r   r<  chordParentaddChordTagr  mxNotechordOrNvelr   graceElementartmxPitchr  	mxTieListmxTiemxVoicemxTypeunused_dotCounterr  mxTimeModificationtupletFraction
tempTupletr  
stem_stylemxStemclosest_clefsdText
nBeamsListmxBmxNotationsListr   tup
tupTagListmxNotationsmxNlyricObjmxLyrics&                                         r9   r^  MeasureExporter.noteToXmlS  sK   r (1,(H"Hv) 	6, x..1133OO22>//0036(CCJJzc#Y0 	v!a//00%fg6L`77m+'Z=V=VW&&2%.#-#?#?A
 ''3%/#-#?#?A 	F&  QWW%>%>$%FJJ.))Cckk)

;. * vw'1gtyy!$Aooagg.GMM'"XXvv&99D ))!,JMM*% 55((/I"e$ # a&&""1k:&$* 1Gt223GL996!1"FK08 	6:>f!&qvvvu% "/ q$))$$GG&&2GG&&44D..qww/A/ABLMM,'qyy>Q!%!>!>qyy|!LMM,-^a&//0K0K0M 22C2CD2I !)C)C)7)A)ACJ
 "&!>!>z!JMM,-  5 x66**m;vvdllH5H$22M q$,,//OO}4t||Q'AOOM
((x~~u??NN,,8!11J$
/F$#+#=#=dii#HH<H)DDXEUEUVF!F&! FKfj1  4
 	f5
 %(DLL11hnn6P!__X^^<
%CMM#& & ..qKP %#AII.3!33CQ?
&&z2 / $V[9K&""3' ' %$OO==(//(3g&	 , 	F#y " `!!$Z[\Z]"^_`^ + a++0I0I!0L0Ls&   ?A.b. c ."ccAd d c                   U R                   b  U R                   R                  c  g[        U R                   R                  5      S::  a  gUR                  (       a  gU(       a  UOUnUR	                  5       nUc  gUnSnU R                   R                   H!  nUR
                  UR
                  :X  d  M  Un  O   Uc  [        SU SU S35      e[        US5      n	UR                  b  U	R                  SUR                  5        gg)z~
Insert <instrument> tags if necessary, that is, when there is more than one
instrument anywhere in the same musicxml <part>.
Nr   zInstrument instance z
 for note z not found in instrumentStreamr   r6  )
r   r  r  r  getInstrumentclassSetr,   r   r  r:  )
ry   r   r  r  searchingObjectclosest_inst_or_noneclosest_instinstance_to_userL  mxInstruments
             r9   r  !MeasureExporter.setNoteInstrument{  s     ;;$++">">"Ft{{++,188CNKTU.<<>'.BKK00D}} 5 55"& 1
 ")&|nJqcA_`  "&,7''3T?#?#?@ 4r;   c                r   U R                  U5      nUR                  S5      nUc  [        S5      e[        R                  " U5      (       ab  UR                  SS5        UR                  S5      nUb  UR                  U5        UR                  S5      nU H  nUR                  U5        M     UR                  S:w  a  [        US5      n[        US	5      nUR                  [        R                  5      n	U	b  [        U	S
5      (       d  [        R                  " 5       n	[        R                   (       a!  [#        U	[        R                  5      (       d   eU	R$                  S-   n
XR                  -   n[&        R(                  " 5       nXl        UR,                  Ul        [1        UR2                  5      Ul        U$ )a  
Convert a Rest object to a <note> with a <rest> tag underneath it.

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> r = note.Rest(quarterLength=2.0)

Give the rest some context:

>>> m = stream.Measure()
>>> m.timeSignature = meter.TimeSignature('4/4')
>>> m.append(r)
>>> mxNoteRest = MEX.restToXml(r)
>>> MEX.dump(mxNoteRest)
<note>
  <rest />
  <duration>20160</duration>
  <type>half</type>
</note>

Now it is a full measure:

>>> m.timeSignature = meter.TimeSignature('2/4')
>>> mxNoteRest = MEX.restToXml(r)
>>> MEX.dump(mxNoteRest)
<note>
  <rest measure="yes" />
  <duration>20160</duration>
</note>

Unless we specify that it should not be converted to a full measure:

>>> r.fullMeasure = False
>>> mxNoteRest = MEX.restToXml(r)
>>> MEX.dump(mxNoteRest)
<note>
  <rest />
  <duration>20160</duration>
  <type>half</type>
</note>

With True or "always" it will be converted to full measure even if
it does not match:

>>> m.timeSignature = meter.TimeSignature('4/4')
>>> r.duration.dots = 1
>>> r.fullMeasure = True
>>> mxNoteRest = MEX.restToXml(r)
>>> MEX.dump(mxNoteRest)
<note>
  <rest measure="yes" />
  <duration>30240</duration>
</note>


Note that if a measure has paddingLeft/paddingRight (such as a pickup)
then a fullMeasure duration might not match the TimeSignature duration.


The display-step and display-octave are set from a Rest's stepShift:

>>> r = note.Rest()
>>> r.stepShift = 1
>>> mxNoteRest = MEX.restToXml(r)
>>> MEX.dump(mxNoteRest)
<note>
  <rest>
    <display-step>C</display-step>
    <display-octave>5</display-octave>
  </rest>
  <duration>10080</duration>
  <type>quarter</type>
</note>

Clef context matters:

>>> m = stream.Measure()
>>> m.clef = clef.BassClef()
>>> m.append(r)
>>> mxNoteRest = MEX.restToXml(r)
>>> MEX.dump(mxNoteRest)
<note>
  <rest>
    <display-step>E</display-step>
    <display-octave>3</display-octave>
  </rest>
  <duration>10080</duration>
  <type>quarter</type>
</note>
ri  z3Something went wrong -- converted rest w/o rest tagr   r  r  rm  r   display-stepdisplay-octave
lowestLiner   )r^  findr,   r(   r  r:  removefindall	stepShiftr   r   r   	PitchClefrt  
TrebleClefr  r  r   r  r   rh   diatonicNoteNumstepr&   rR   octave)ry   r  r  	mxRestTagr  mxDotsmxDotmxDisplayStepmxDisplayOctavecurrentClef
midLineDNNrestObjectPseudoDNN	tempPitchs                r9   rk  MeasureExporter.restToXml  sa   v "KK'	)*_``$$Q''MM)U+[[(F!f%^^E*Fe$   ;;!&y.AM(4DEO/0/B/B4>>/RK"'+|*L*L"oo/ !+t~~>>>>$//!3J",{{":I(;%!*M#&y'7'7#8O r;   c           	     0   / n[        U[        R                  5      (       a  UR                  5         [	        U5       HU  u  p4SUR
                  ;   a!  UR                  U R                  XCUS95        M6  UR                  U R                  XCUS95        MW     U$ )av  
Returns a list of <note> tags, all but the first with a <chord/> tag on them.
And appends them to self.xmlRoot

Attributes of notes are merged from different locations: first from the
duration objects, then from the pitch objects. Finally, GeneralNote
attributes are added.

>>> ch = chord.Chord()
>>> ch.quarterLength = 2
>>> b = pitch.Pitch('A-2')
>>> c = pitch.Pitch('D3')
>>> d = pitch.Pitch('E4')
>>> e = [b, c, d]
>>> ch.pitches = e

>>> len(ch.pitches)
3
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> len(MEX.xmlRoot)
0
>>> mxNoteList = MEX.chordToXml(ch)
>>> len(mxNoteList)  # get three mxNotes
3
>>> len(MEX.xmlRoot)
3

>>> MEX.dump(mxNoteList[0])
<note>
  <pitch>
    <step>A</step>
    <alter>-1</alter>
    <octave>2</octave>
  </pitch>
  <duration>20160</duration>
  <type>half</type>
  <accidental>flat</accidental>
</note>

>>> MEX.dump(mxNoteList[1])
<note>
  <chord />
  <pitch>
    <step>D</step>
    <octave>3</octave>
  </pitch>
  <duration>20160</duration>
  <type>half</type>
</note>

>>> MEX.dump(mxNoteList[2])
<note>
  <chord />
  <pitch>
    <step>E</step>
    <octave>4</octave>
  </pitch>
  <duration>20160</duration>
  <type>half</type>
</note>


Test that notehead and style translation works:

>>> g = pitch.Pitch('g3')
>>> h = note.Note('b4')
>>> h.notehead = 'diamond'
>>> h.style.color = 'gold'
>>> h.style.absoluteX = 176
>>> ch2 = chord.Chord([g, h])
>>> ch2.quarterLength = 2.0
>>> mxNoteList = MEX.chordToXml(ch2)
>>> MEX.dump(mxNoteList[1])
<note color="#FFD700" default-x="176">
  <chord />
  <pitch>
    <step>B</step>
    <octave>4</octave>
  </pitch>
  <duration>20160</duration>
  <type>half</type>
  <notehead color="#FFD700" parentheses="no">diamond</notehead>
</note>

And unpitched chord members:

>>> perc = percussion.PercussionChord([note.Unpitched(), note.Unpitched()])
>>> for n in MEX.chordToXml(perc):
...     MEX.dump(n)
<note>
  <unpitched>
    <display-step>B</display-step>
    <display-octave>4</display-octave>
  </unpitched>
  <duration>10080</duration>
  <type>quarter</type>
</note>
<note>
  <chord />
  <unpitched>
    <display-step>B</display-step>
    <display-octave>4</display-octave>
  </unpitched>
  <duration>10080</duration>
  <type>quarter</type>
</note>

Test articulations of chords with fingerings. Superfluous fingerings will be ignored.

>>> testChord = chord.Chord('E4 C5')
>>> testChord.articulations = [articulations.Fingering(1),
...        articulations.Accent(), articulations.Fingering(5), articulations.Fingering(3)]
>>> for n in MEX.chordToXml(testChord):
...     MEX.dump(n)
<note>
  ...
  <notations>
    <articulations>
      <accent />
    </articulations>
    <technical>
      <fingering alternate="no" substitution="no">1</fingering>
    </technical>
  </notations>
</note>
<note>
  <chord />
  ...
  <notations>
    <technical>
      <fingering alternate="no" substitution="no">5</fingering>
    </technical>
  </notations>
</note>
ri  r<  r  )	r   r   r[  sortAscendingrC  r  r   rj  r^  )ry   r  
mxNoteListr   r   s        r9   rh  MeasureExporter.chordToXml  s    R 
a%%OOaLDAajj(!!$"5"5aYZ"5"[\!!$..TU."VW	 !
 r;   c                    [        S5      n[        [        [        U R                  UR
                  -  5      5      5      Ul        U$ )a  
Convert a duration.Duration object to a <duration> tag using self.currentDivisions

>>> d = duration.Duration(1.5)
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> MEX.currentDivisions = 10
>>> mxDuration = MEX.durationXml(d)
>>> MEX.dump(mxDuration)
<duration>15</duration>
r   )r   rR   r7  r  r  r   r&   )ry   durr  s      r9   r~  MeasureExporter.durationXml  s;     Z(
c%(=(=@Q@Q(Q"RST
r;   c                    [        S5      n[        XSS5        UR                  bD  [        US5      n[	        [
        R                  " UR                  R                  5      5      Ul        [        XSS5        U$ )a!  
Convert a :class:`~music21.pitch.Pitch` to xml.
Does not create the <accidental> tag.

>>> p = pitch.Pitch('D#5')
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxPitch = MEX.pitchToXml(p)
>>> MEX.dump(mxPitch)
<pitch>
  <step>D</step>
  <alter>1</alter>
  <octave>5</octave>
</pitch>
r   r  alterr  implicitOctave)	r   rX   r  r   rR   r   numToIntOrFloatr  r&   )ry   r   r  mxAlters       r9   r|  MeasureExporter.pitchToXml  sc      '" VV<<<# '2Gv55all6H6HIJGL X7GHr;   c                    U R                  XUS9n[        S5      n[        XS5        [        XS5        [        R                  " XESS/S9  U$ )a  
Convert an :class:`~music21.note.Unpitched` to a <note>
with an <unpitched> subelement.

>>> up = note.Unpitched(displayName='D5')
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxUnpitched = MEX.unpitchedToXml(up)
>>> MEX.dump(mxUnpitched)
<note>
  <unpitched>
    <display-step>D</display-step>
    <display-octave>5</display-octave>
  </unpitched>
  <duration>10080</duration>
  <type>quarter</type>
</note>

>>> graceUp = up.getGrace()
>>> mxUnpitched = MEX.unpitchedToXml(graceUp)
>>> MEX.dump(mxUnpitched)
<note>
  <grace slash="yes" />
  <unpitched>
    <display-step>D</display-step>
    <display-octave>5</display-octave>
  </unpitched>
  <type>quarter</type>
</note>
r  	unpitchedr  r  r   r  )tagList)r^  r   rX   r(   insertBeforeElements)ry   rp  r<  r  r  mxUnpitcheds         r9   rj  MeasureExporter.unpitchedToXml  sQ    D S^_k* .A 2BC$$V:vBVWr;   c                    [        S5      n[        XSS5        [        XSS5        UR                  b  [        XSS5        U$ )a  
Converts a FretNote Object to MusicXML readable format.

Note that, although music21 is referring to FretNotes as FretNotes,
musicxml refers to them as frame notes. To convert between the two formats,
each 'Fret-Note' must be converted to 'Frame-Note'.

>>> fn = tablature.FretNote(string=3, fret=1, fingering=2)
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> MEXFretNote = MEX.fretNoteToXml(fn)
>>> MEX.dump(MEXFretNote)
<frame-note>
    <string>3</string>
    <fret>1</fret>
    <fingering>2</fingering>
</frame-note>

Without fingering!

>>> fn2 = tablature.FretNote(string=5, fret=2)
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> MEXOtherFretNote = MEX.fretNoteToXml(fn2)
>>> MEX.dump(MEXOtherFretNote)
<frame-note>
    <string>5</string>
    <fret>2</fret>
</frame-note>
z
frame-notestringfret	fingering)r   rX   r  )ry   fretNotemxFrameNotes      r9   fretNoteToXmlMeasureExporter.fretNoteToXml  sD    : l+ (K G)$XKUr;   c                J   UR                   (       d  g[        S5      n[        US5      n[        UR                  5      Ul        [        US5      n[        UR                  5      Ul        UR                  5        H%  nU R                  U5      nUR                  U5        M'     U$ )a^  
The ChordWithFretBoard Object combines chord symbols with FretNote objects.

>>> myFretNote1 = tablature.FretNote(1, 2, 2)
>>> myFretNote2 = tablature.FretNote(2, 3, 3)
>>> myFretNote3 = tablature.FretNote(3, 2, 1)
>>> guitarChord = tablature.ChordWithFretBoard('DM', numStrings=6,
...                    fretNotes=[myFretNote1, myFretNote2, myFretNote3])
>>> guitarChord.tuning = tablature.GuitarFretBoard().tuning
>>> guitarChord.getPitches()
[None, None, None,
 <music21.pitch.Pitch A3>, <music21.pitch.Pitch D4>, <music21.pitch.Pitch F#4>]
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> MEXChordWithFret = MEX.fretBoardToXml(guitarChord)
>>> MEX.dump(MEXChordWithFret)
<frame>
    <frame-strings>6</frame-strings>
    <frame-frets>4</frame-frets>
    <frame-note>
        <string>3</string>
        <fret>2</fret>
        <fingering>1</fingering>
    </frame-note>
    <frame-note>
        <string>2</string>
        <fret>3</fret>
        <fingering>3</fingering>
    </frame-note>
    <frame-note>
        <string>1</string>
        <fret>2</fret>
        <fingering>2</fingering>
    </frame-note>
</frame>
Nframezframe-stringszframe-frets)
	fretNotesr   r   rR   
numStringsr&   displayFretsfretNotesLowestFirstr  r   )ry   	fretBoardmxFramemxFrameStringsmxFrameFretsthisFretNote
mxFretNotes          r9   fretBoardToXmlMeasureExporter.fretBoardToXml-  s    H "" '"#G_=!)"6"67!'=9	 6 67%::<L++L9JNN:& = r;   c                    U R                  U5      n[        R                  (       a  [        U[        5      (       d   eU R                  U5      nUb  UR                  U5        U$ )zJ
Deals with both chords and frets.
Generate harmony and append xml to it.
)rd  r  r  r   r   r  r   )ry   cwf	mxHarmonyr  s       r9   rb  'MeasureExporter.chordWithFretBoardToXmla  sX    
 ))#.	??i1111%%c*W%r;   c                b   [        S5      n[        XSS5        [        XSS5        UR                  b|  [        US5      n[	        UR                  R
                  5      Ul        UR                  R                  S:  a2  [        UR                  R                  5       H  n[        US5        M     U$ )	a.  
>>> tup = duration.Tuplet(11, 8)
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxTimeMod = MEX.tupletToTimeModification(tup)
>>> MEX.dump(mxTimeMod)
<time-modification>
  <actual-notes>11</actual-notes>
  <normal-notes>8</normal-notes>
</time-modification>

>>> tup.setDurationType('eighth', dots=2)
>>> mxTimeMod = MEX.tupletToTimeModification(tup)
>>> MEX.dump(mxTimeMod)
<time-modification>
  <actual-notes>11</actual-notes>
  <normal-notes>8</normal-notes>
  <normal-type>eighth</normal-type>
  <normal-dot />
  <normal-dot />
</time-modification>

ztime-modificationzactual-notesnumberNotesActualznormal-notesnumberNotesNormalznormal-typer   z
normal-dot)	r   rX   durationNormalr   r:   r  r&   r  r   )ry   r  r  mxNormalTyper   s        r9   r  (MeasureExporter.tupletToTimeModificationr  s    0 %%89 .J]^ .J]^)%&8-HL 233E3E3J3J KL!!&&*s11667A1<@ 8 "!r;   c                   Sn[        U[        R                  5      (       a  UR                  S:w  dI  UR                  (       d8  UR
                  c+  UR                  (       ac  UR                  R                  S;  aI  [        R                  " [        R                  U5      nSnU R                  U5      nUR                  U5        USL a  Ub  [        US5      (       a~  UR                  S:w  dI  UR                  (       d8  UR
                  c+  UR                  (       a>  UR                  R                  S;  a#  U R                  U5      nUR                  U5        gggggg)aM  
Determine if an <notehead> element needs to be added to this <note>
element (mxNote) and if it does then get the <notehead> element from
noteheadToXml and add it to mxNote.

Complicated because the chordParent might have notehead
set, which would affect every note along the way.

Returns nothing.  The mxNote is modified in place.
FrX  NrN   Tnotehead)r   r   rs  r  noteheadParenthesisnoteheadFillr4  r$   rB   r  r{  noteheadToXmlr   rt  )ry   r  r   r  foundANotehead
mxNoteheads         r9   r   MeasureExporter.dealWithNotehead  s     q$,,''x'))NN.**qww}}J/Nt||Q'A!N++A.JMM*%U"{'>Z00 ))X5#77#00<$88(..44JF!//<
j) G 9	 1 (?"r;   c                l   [        S5      nUR                  nUc  SnX2l        [        nU" XSS[        R
                  S9  U" XSS[        R
                  S9  UR                  (       aL  UR                  R                  (       a1  [        UR                  R                  5      nUR                  SU5        U$ )	aG  
Translate a music21 :class:`~music21.note.NotRest` object
such as a Note, or Chord into a `<notehead>` tag.

>>> n = note.Note('C#4')
>>> n.notehead = 'diamond'
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxN = MEX.noteheadToXml(n)
>>> MEX.dump(mxN)
<notehead parentheses="no">diamond</notehead>

>>> n1 = note.Note('c3')
>>> n1.style.color = 'red'
>>> n1.notehead = 'diamond'
>>> n1.noteheadParenthesis = True
>>> n1.noteheadFill = False
>>> mxN = MEX.noteheadToXml(n1)
>>> MEX.dump(mxN)
<notehead color="#FF0000" filled="no" parentheses="yes">diamond</notehead>

>>> n1 = note.Note('c3')
>>> n1.style.color = 'red'
>>> n1.notehead = 'diamond'
>>> n1.noteheadParenthesis = True
>>> n1.noteheadFill = False
>>> mxN = MEX.noteheadToXml(n1)
>>> MEX.dump(mxN)
<notehead color="#FF0000" filled="no" parentheses="yes">diamond</notehead>
r  rr  filledr  r  r  r  rB   )r   r  r&   r  r*   r8  r4  r$   rB   rC   r:  )ry   r   r  nhr  rB   s         r9   r  MeasureExporter.noteheadToXml  s    > Z(
ZZ:B(QHn
@Y@YZQM+@!00	2   QWW]]"177==1ENN7E*r;   c                $   S nUR                   S:X  al  US:X  a%  U R                  U5      nUR                  SS5        U$ U[        UR                  5      S-
  :X  a#  U R                  U5      nUR                  SS5        U$ U R                  U5      nU$ )NrU  r   r  r  r   r  )r  expressionToXmlr:  r  notes)ry   arpeggioMarkchordOrNoter<  mxExpressions        r9   arpeggioMarkToMxExpression*MeasureExporter.arpeggioMarkToMxExpression  s    .1$#33LA  2  "S):):%;a%??#33LA  /   //=Lr;   c                   / n[        U[        R                  5      (       Ga2  UR                  (       a  UR                  R                  (       al  UR                  R                  R
                  SL aI  UR                  U R                  UR                  R                  5      5        US   R                  SS5        UR                  (       a  UR                  R                  (       al  UR                  R                  R
                  SL aI  UR                  U R                  UR                  R                  5      5        US   R                  SS5        U$ [        U[        R                  [        R                  45      (       a  UR                  (       a  UR                  R                  (       a  UR                  R                  R
                  SL ap  UR                  U R                  UR                  R                  5      5        UR                  S:X  a  US   R                  SS5        U$ US   R                  SS5        U$ )NTr&  r  abovebelowdown)r   r   TurnupperOrnamentalPitchr  r  r   r  r:  lowerOrnamentalPitchGeneralMordentTrillornamentalPitchr  )ry   ornamentObjmxAccidentalMarkss      r9   ornamentToMxAccidentalMarks+MeasureExporter.ornamentToMxAccidentalMarks  s   +-k;#3#34400#88CC#88CCQQUYY!((55k6V6V6a6ab ""%))+w?00#88CC#88CCQQUYY!((55k6V6V6a6ab ""%))+w? !  k&@&@+BSBS%TUU++#33>>#33>>LLPTT!((55k6Q6Q6\6\] ((F2%b)--k7C !  &b)--k7C  r;   c                   SnSnSnUS:H  n/ nUn	Ub  Un	U	R                    H  n
[        U
[         R                  5      (       a*  U R                  XU5      nUc  M9  UR	                  U5        ML  U(       d  MU  U R                  U
5      nUc  Mk  [        U
[         R                  5      (       aL  Uc  [        S5      nUR	                  U5        U R                  U
5      nU H  nUR	                  U5        M     M  UR	                  U5        M     / nSnU	R                   H  n[        U[        R                  5      (       a  X:X  a  UR	                  U5        US-  nM?  U(       d  MH  [        U[        R                  [        R                  45      (       a  My  UR	                  U5        M     U H  n[        U[        R                  5      (       a  M$  [        U[        R                  5      (       a  UR                  S:  a  MU  [        U[        R                   5      (       a0  Uc  [        S5      nUR	                  U R#                  U5      5        M  Uc  [        S5      nUR	                  U R%                  U5      5        M     UR&                  b,  U R)                  UR&                  5      nUR+                  U5        Ub!  UR+                  U R-                  X25      5        O UR+                  U R-                  X5      5        U(       a  Ub  U R/                  U5      nOUb  SnOU R/                  U5      nU(       a  Uc  [        S5      nUR+                  U5        UUU4 H  nUc  M  UR	                  U5        M     U$ )zj
Take information from .expressions,
.articulations, and spanners to
make the <notations> tag for a note.
Nr   r6  r   	technicalr   )r   r   ArpeggioMarkr  r   r  Ornamentr   r*  r   	FingeringrM  rN  rg  StringIndicationr   TechnicalIndicationarticulationToXmlTechnicalarticulationToXmlArticulationr'   tieToXmlTiedr  rH  rR  )ry   r   r<  r  mxArticulationsmxTechnicalMarkmxOrnamentsr?  r=  r  expObjr  mxAccidMarksmxAccidMarkapplicableArticulationsfingeringNumberaObjtiedListspannerTechnicalsr0  s                       r9   r  MeasureExporter.noteToNotations  s9    &6!&;"#%	"%K "--F&+":":;;#>>)9   '  .++#33F;' fk&:&:;;"*&-k&:&&|4 88@ ! (4#**;7 (4 $$\29 .@ EG--D$ 7 788"6+22481$++ "$)?)?AVAV(WXX+2248 . ,D$ 7 788$ > >??DKKRSO$ A ABB"*&-k&:O&&t'F'Ft'LM"*&-o&>O&&t'I'I$'OP ,( 55((/HX& "TCCKbcTCCAXY
 &+*A $ G G T$ $ $ G G J&")+"6""#45!!A }  #	 r;   c                    / nUR                   S:X  a  SnOUR                   n[        S5      nUR                  SU5        UR                  U5        UR                   S:X  a.  [        S5      nUR                  SS5        UR                  U5        U$ )a0  
returns a list of ties from a Tie object.

A 'continue' tie requires two <tie> tags to represent.

>>> tieObj = tie.Tie('continue')
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> tieList = MEX.tieToXmlTie(tieObj)
>>> for mxT in tieList:
...     MEX.dump(mxT)
<tie type="stop" />
<tie type="start" />
continuer  r'   r  r  )r  r   r:  r   )ry   m21Tier  musicxmlTieTyper  s        r9   r  MeasureExporter.tieToXmlTie  s     	;;*$$O$kkO		&/*;;*$ENEIIfg&U#r;   c                   UR                   S:X  a  / $ / nUR                  S:X  a  SnOUR                  n[        S5      n[        XA5        UR	                  SU5        UR                  U5        UR                  S:X  a.  [        S5      nUR	                  SS5        UR                  U5        UR                   S:w  a,  UR                  S:w  a  UR	                  SUR                   5        UR                  b_  UR	                  S
UR                  5        S	nUR                  S:X  a  SnOUR                  S:X  a  SnOSnU(       a  UR	                  SU5        U$ )a^  
In musicxml, a tie is represented in sound
by the tie tag (near the pitch object), and
the <tied> tag in notations.  This
creates the <tied> tag.

Returns a list since a music21 "continue" tie type needs two tags
in musicxml.  List may be empty if tie.style == "hidden"

>>> tieObj = tie.Tie('continue')
>>> tieObj.id = 'tied1'

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> tiedList = MEX.tieToXmlTied(tieObj)
>>> for mxT in tiedList:
...     MEX.dump(mxT)
<tied id="tied1" type="stop" />
<tied type="start" />

>>> tieObj.style = 'hidden'
>>> tiedList = MEX.tieToXmlTied(tieObj)
>>> len(tiedList)
0
hiddenrC  r  tiedr  r  rX  r  Nr  r  overr   underr=   orientation)r$   r  r   r  r:  r   r  )ry   rD  
mxTiedListrE  mxTiedrL  s         r9   r5  MeasureExporter.tieToXmlTied  s*   4 <<8#I
;;*$$O$kkOv&

6?+&!;;*$V_FJJvw'f% <<8#v(=JJ{FLL1 'JJ{F$4$45K7*$!!W,% 
 

=+6
 r;   c                `   UR                   S;   a  / $ UR                   S;  a  [        SUR                   -   5      eUR                   S:X  a  SS/nOUR                   /n/ nU GH  n[        S5      nUR                  SU5        UR                  S	[	        U5      5        US:X  Gat  UR
                  nUS
:X  a  SnUR                  S[        R                  " U5      5        UR                  b  UR                  SUR                  5        UR                  nUR                  n	Uc  UR                  SS5        O7US;   a  U	S;   a  UR                  SS5        OUS;   a  UR                  SS5        US;   a  U	S;   a  UR                  SS5        OUS;   a  UR                  SS5        UR
                  S
:X  a  UR                  SS5        [        US5      n
[        U
S5      n[	        UR                  5      Ul        UR                  bj  [        UR                  R                   5      nU(       a  [        U
S5      nXl        [!        UR                  R"                  5       H  n[        U
S5        M     [        US5      n[        US5      n[	        UR$                  5      Ul        UR&                  bk  [        UR&                  R                   5      nU(       a  [        US5      nUUl        [!        UR&                  R"                  5       H  n[        US5        M     UR)                  U5        GM     U$ )a  
In musicxml, a tuplet is represented by
a timeModification and visually by the
<notations><tuplet> tag.  This method
creates the latter.

Returns a list of them because a
startStop type tuplet needs two tuplet
brackets.

TODO: make sure something happens if
    makeTupletBrackets is not set.

>>> tup = duration.Tuplet(11, 8)
>>> tup.type = 'start'
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxTup = MEX.tupletToXmlTuplet(tup)
>>> len(mxTup)
1
>>> MEX.dump(mxTup[0])
<tuplet bracket="yes" number="1" placement="above" type="start">
  <tuplet-actual>
    <tuplet-number>11</tuplet-number>
  </tuplet-actual>
  <tuplet-normal>
    <tuplet-number>8</tuplet-number>
  </tuplet-normal>
</tuplet>

>>> tup.tupletActualShow = 'both'
>>> tup.tupletNormalShow = 'type'
>>> mxTup = MEX.tupletToXmlTuplet(tup)
>>> MEX.dump(mxTup[0])
<tuplet bracket="yes" number="1" placement="above"
    show-number="actual" show-type="both" type="start">...</tuplet>
)NFr=   )r  r  	startStopz.Cannot create music XML from a tuplet of type rQ  r  r  tupletr  r   r.  Tr  r  zshow-numberrr  )r  r   r  )r  actualrS  )r  r  z	show-type
line-shapecurvedztuplet-actualztuplet-numberztuplet-typez
tuplet-dotztuplet-normal)r  r,   r   r:  rR   r  r*   r8  r  tupletActualShowtupletNormalShowr   r  r&   durationActualr:   r   r  r  r  r   )ry   rR  tupletIndex	localTyperetList
tupletTypemxTuplettBrackettastnsmxTupletActualmxTupletNumber
actualTypemxTupletTypeunused_countermxTupletNormal
normalTypes                    r9   r  !MeasureExporter.tupletToXmlTuplet  s   J ;;++I;;<<)@6;;NP P ;;+% &)II#Jx(HLL,LL3{#34W$!>>v%#HY'66x@B##/LLf.>.>?----;LL7..3:L3LLL7..LL9**s6F/FLLf5,,LLh7>>V+LLx8 ",Ho!F!+NO!L&)&*B*B&C#((4!3F4I4I4N4N!OJ!'1.-'P,6)*/0E0E0J0J*K"><@ +L ",Ho!F!+NO!L&)&*B*B&C#((4!3F4I4I4N4N!OJ!'1.-'P,6)*/0E0E0J0J*K"><@ +L NN8$o $p r;   c                   SnUR                   n[        U[        R                  5      (       aS  UR                  S:X  a  [        S5      nO7[        S5      nUR                  S:w  a  UR                  SUR                  5        [        U[        R                  5      (       ap  [        U[        R                  5      (       a)  UR                  (       a  [        S5      nO4[        S5      nO(UR                  (       a  [        S	5      nO[        S
5      nUc8  [        / SQ5      nUR                  5        H  u  pVXS;   d  M  [        U5      n  O   Uc"  [        R                  SU/5        [        S5      $ U R                  X!5        [        US5      (       a)  UR                   b  UR                  SUR                   5        [        U[        R"                  5      (       aQ  UR                  S[%        UR                  5      5        UR&                  S;   a  UR&                  Ul        [+        X!5        [        U[        R,                  5      (       a,  UR                  SS5        [%        UR.                  5      Ul        U$ )a  
Convert a music21 Expression (expression or ornament)
to a musicxml tag;
return None if no conversion is possible.

Expressions apply only to the first note of chord.

>>> invTurn = expressions.InvertedTurn()
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxExpression = MEX.expressionToXml(invTurn)
>>> MEX.dump(mxExpression)
<inverted-turn placement="above" />

>>> invDelayedTurn = expressions.InvertedTurn(delay=1.0)
>>> invDelayedTurn.placement = 'below'
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxExpression = MEX.expressionToXml(invDelayedTurn)
>>> MEX.dump(mxExpression)
<delayed-inverted-turn placement="below" />

Some special types:

>>> f = expressions.Fermata()
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxExpression = MEX.expressionToXml(f)
>>> MEX.dump(mxExpression)
<fermata type="inverted" />
>>> f.shape = 'angled'
>>> mxExpression = MEX.expressionToXml(f)
>>> MEX.dump(mxExpression)
<fermata type="inverted">angled</fermata>

>>> trem = expressions.Tremolo()
>>> trem.numberOfMarks = 4
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxExpression = MEX.expressionToXml(trem)
>>> MEX.dump(mxExpression)
<tremolo type="single">4</tremolo>

>>> arp = expressions.ArpeggioMark()
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxExpression = MEX.expressionToXml(arp)
>>> MEX.dump(mxExpression)
<arpeggiate />
>>> arp.type = 'down'
>>> mxExpression = MEX.expressionToXml(arp)
>>> MEX.dump(mxExpression)
<arpeggiate direction="down" />

>>> nonArp = expressions.ArpeggioMark('non-arpeggio')
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxExpression = MEX.expressionToXml(nonArp)
>>> MEX.dump(mxExpression)
<non-arpeggiate />
NrU  rV  rW  rX  r  zdelayed-inverted-turnzinverted-turnzdelayed-turnturn))r&  z
trill-mark)InvertedMordentzinverted-mordent)Mordentmordent)Shakeshake)	Schleifer	schleifer)Tremolor4  )Fermatafermata)r/  other-ornamentzno musicxml conversion for:ru  r  r  )angledsquaresingle)r   r   r   r.  r  r   r:  r"  InvertedTurn	isDelayedr   r   r  r  rX  rt  r  rs  rR   shaper&   r  rr  r;  )ry   
expressionmxr   mappingr  r   s          r9   r  MeasureExporter.expressionToXmlt  s   p $$ j+":":;;.0-.\*??h.FF;
8 j+"2"233*k&>&>??'' !89B 1B'' 0B B:! # G  < B (
 :##%BJ$OP+,,2*:{++
0D0D0PFF;
 4 45j+"5"566FF63z/0#77$**2*j+"5"566FF68$*223BG	r;   c                   Sn[         R                   H(  n[        X5      (       d  M  [         R                  U   n  O   Uc  Sn[        U5      nUR                  b  UR                  SUR                  5        U R                  XA5        US:X  aR  [        R                  (       a!  [        U[        R                  5      (       d   eUR                  SUR                  5        US;   a  U R                  XA5        US:X  aT  [        R                  (       a!  [        U[        R                  5      (       d   eUR                  b  UR                  Ul        US:X  a  UR                   b  UR                   Ul        U$ )a  
Returns a class (mxArticulationMark) that represents the
MusicXML structure of an articulation mark.

>>> a = articulations.Accent()
>>> a.placement = 'below'

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxArticulationMark = MEX.articulationToXmlArticulation(a)
>>> MEX.dump(mxArticulationMark)
<accent placement="below" />

>>> a = articulations.Staccatissimo()
>>> a.placement = 'below'

>>> mxArticulationMark = MEX.articulationToXmlArticulation(a)
>>> MEX.dump(mxArticulationMark)
<staccatissimo placement="below" />

Articulations can have style attached to them:

>>> a = articulations.Doit()
>>> a.style.lineShape = 'curved'
>>> a.style.lineType = 'dashed'
>>> a.style.dashLength = 2
>>> a.style.spaceLength = 1
>>> a.style.absoluteX = 5
>>> a.style.absoluteY = 2

>>> mxArticulationMark = MEX.articulationToXmlArticulation(a)
>>> MEX.dump(mxArticulationMark)
<doit dash-length="2" default-x="5" default-y="2"
    line-shape="curved" line-type="dashed" space-length="1" />
Nzother-articulationr  zstrong-accentr  )doitfalloffplopscoopzbreath-mark)r*   ARTICULATION_MARKS_REVr   r   r  r:  rX  r  r  r   StrongAccentpointDirectionr8  
BreathMarkr  r&   displayText)ry   articulationMarkmusicXMLArticulationNamer  mxArticulationMarks        r9   r4  -MeasureExporter.articulationToXmlArticulation  sM   P $( 22A*..+5+L+LQ+O( 3 $+';$$%=>%%1"";0@0J0JK-@#6!"2M4N4NOOOO""6+;+J+JK#'KK0C#}4!"2M4L4LMMMM&&2*:*A*A"'$(<<$00<&6&B&B#!!r;   c                0    SnSnU R                  XX45        g)z
Sets four additional elements for line elements, conforms to entity
%line-shape, %line-type, %dashed-formatting (dash-length and space-length)
)rT  r  zdash-lengthzspace-length)	lineShaper  
dashLengthspaceLengthNr}  rP  s        r9   r8  MeasureExporter.setLineStyle2  s    
 SI]Mr;   c                F   Sn[         R                   H(  n[        X5      (       d  M  [         R                  U   n  O   Uc  SnUS;   a  Sn[        U5      nUR                  b  UR                  SUR                  5        US:X  a  [        R                  (       a!  [        U[        R                  5      (       d   e[        UR                  5      Ul        UR                  S[         R                  " UR                  5      5        US;   a  UR                  b  UR                  Ul        US;   av  [        R                  (       a1  [        U[        R                  [        R                   45      (       d   eUR                  S	[         R                  " UR"                  5      5        US
:X  aP  [        R                  (       a!  [        U[        R$                  5      (       d   e[        UR&                  5      Ul        US:X  aP  [        R                  (       a!  [        U[        R(                  5      (       d   e[        UR&                  5      Ul        US:X  aG  [        R                  (       a!  [        U[        R*                  5      (       d   eU R-                  XA5        US:X  aG  [        R                  (       a!  [        U[        R.                  5      (       d   eU R1                  XA5        US:X  a  UR                  b  UR                  Ul        U R3                  XA5        U$ )a   
Returns a tag that represents the
MusicXML structure of an articulation mark that is primarily a TechnicalIndication.

>>> MEX = musicxml.m21ToXml.MeasureExporter()

>>> a = articulations.UpBow()
>>> a.placement = 'below'

>>> mxTechnicalMark = MEX.articulationToXmlTechnical(a)
>>> MEX.dump(mxTechnicalMark)
<up-bow placement="below" />


>>> f = articulations.Fingering(4)
>>> f.substitution = True
>>> mxFingering = MEX.articulationToXmlTechnical(f)
>>> MEX.dump(mxFingering)
<fingering alternate="no" substitution="yes">4</fingering>

Technical marks too specific to express in musicxml just get other-technical.

>>> g = articulations.OrganIndication()
>>> g.displayText = 'unda maris'
>>> mxOther = MEX.articulationToXmlTechnical(g)
>>> MEX.dump(mxOther)
<other-technical>unda maris</other-technical>

Same with technical marks not yet supported.
TODO: support HammerOn, PullOff, Hole, Arrow.

>>> h = articulations.HammerOn()
>>> mxOther = MEX.articulationToXmlTechnical(h)
>>> MEX.dump(mxOther)
<other-technical />
Nother-technical)holearrowr  r  	alternate)handbellr  )heeltoer  substitutionr  r  bendharmonic)r*   TECHNICAL_MARKS_REVr   r   r  r:  r  r  r   r0  rR   fingerNumberr&   r8  r  r  OrganIndicationr  r1  r   FretIndicationFretBendsetBendStringHarmonicsetHarmonicrX  )ry   r  musicXMLTechnicalNamer  r7  s        r9   r3  *MeasureExporter.articulationToXmlTechnicalH  s   V !%//A*..(2(F(Fq(I% 0 !($5! !$55$5!!"78%%1-=-G-GH K/!"2M4K4KLLLL#&'7'D'D#EO  * 9 9:J:T:T UW!%DD$00< $4#?#?O  $@@!"2]5L5L5B5R5R5T U U U U * 9 9:J:W:W XZ H,!"2M4R4RSSSS#&'7'>'>#?O  F*!"2M4P4PQQQQ#&'7'>'>#?O  F*!"2M4J4JKKKKLL; !J.!"2M4P4PQQQQ_?!%66$00<#3#?#?O ?=r;   c                   [        U S5      nUR                  nUb  [        UR                  5      Ul        UR
                  (       a  [        U S5        UR                  bO  [        U S5      nUR                  n[        [        R                  U-  5      nUR                  S[        U5      5        UR                  b'  [        U S5      n[        UR                  5      Ul        gg)a}  
Sets the bend-alter SubElement and the pre-bend,
release and with-bar SubElements when present.

Called from articulationToXmlTechnical

>>> from xml.etree.ElementTree import Element
>>> from fractions import Fraction

>>> MEXclass = musicxml.m21ToXml.MeasureExporter

>>> a = articulations.FretBend(bendAlter=interval.Interval(2))
>>> mxh = Element('bend')
>>> MEXclass.setBend(mxh, a)
>>> MEXclass.dump(mxh)
<bend>
  <bend-alter>2</bend-alter>
</bend>
>>> mxh = Element('bend')
>>> a = articulations.FretBend(bendAlter=interval.Interval(2))
>>> a.preBend = True
>>> MEXclass.setBend(mxh, a)
>>> MEXclass.dump(mxh)
<bend>
  <bend-alter>2</bend-alter>
  <pre-bend />
</bend>
>>> mxh = Element('bend')
>>> a = articulations.FretBend(bendAlter=interval.Interval(-2), release=Fraction(1, 10080))
>>> MEXclass.setBend(mxh, a)
>>> MEXclass.dump(mxh)
<bend>
  <bend-alter>-2</bend-alter>
  <release offset="1" />
</bend>
>>> mxh = Element('bend')
>>> a = articulations.FretBend(bendAlter=interval.Interval(-2), withBar='scoop')
>>> MEXclass.setBend(mxh, a)
>>> MEXclass.dump(mxh)
<bend>
  <bend-alter>-2</bend-alter>
  <with-bar>scoop</with-bar>
</bend>
z
bend-alterNzpre-bendreleaser  zwith-bar)r   	bendAlterrR   	semitonesr&   preBendr  r7  r   r  r:  withBar)mxhr  bendAlterSubElementr  releaseSubElementquarterLengthValuedivisionsValuewithBarSubElements           r9   r  MeasureExporter.setBend  s    \ )l;'*5??';$<<sJ'<<# *3	 :!% !=!=@R!RSN!!(C,?@<<# *3
 ;%(%6" $r;   c                H   [        US5      (       d  gUR                  S:X  a  [        U S5        OUR                  S:X  a  [        U S5        UR                  S:X  a  [        U S5        gUR                  S:X  a  [        U S5        gUR                  S	:X  a  [        U S
5        gg)a  
Sets the artificial or natural tag (or no tag) and
zero or one of base-pitch, sounding-pitch, touching-pitch

Called from articulationToXmlTechnical

>>> MEXClass = musicxml.m21ToXml.MeasureExporter

>>> a = articulations.StringHarmonic()
>>> a.harmonicType = 'artificial'
>>> a.pitchType = 'sounding'

>>> from xml.etree.ElementTree import Element
>>> mxh = Element('harmonic')

>>> MEXClass.setHarmonic(mxh, a)
>>> MEXClass.dump(mxh)
<harmonic>
  <artificial />
  <sounding-pitch />
</harmonic>

:class:`~music21.articulations.Harmonic` is probably too general for most uses,
but if you use it, you get a harmonic tag with no details:

>>> b = articulations.Harmonic()
>>> mxh2 = Element('harmonic')
>>> MEXClass.setHarmonic(mxh2, b)
>>> MEXClass.dump(mxh2)
<harmonic />
harmonicTypeN
artificialnaturalr	   z
base-pitchsoundingzsounding-pitchtouchingztouching-pitch)rt  r  r   	pitchType)r  harms     r9   r  MeasureExporter.setHarmonic  s    D t^,,,sL))+sI&>>V#sL)^^z)s,-^^z)s,- *r;   c                   UR                   SL a/  [        R                  " UR                  S9nU R	                  U5      $ [        S5      n[        X15        U R                  X15        U R                  X15        [        US5      n[        US5      nSUl
        UR                  SS5        [        US	5      nUR                  nUR                  S
:w  a  [        S5      e[        U5      Ul
        UR                  S;   a  [        S5      eUR                  SUR                  5        U R!                  X5        U R#                  X15        U R$                  R'                  U5        U$ )a  
Convert a NoChord object to an mxHarmony object (or a rest
if .writeAsChord = True).

Expected attributes of the NoChord object:

>>> nc = harmony.NoChord()
>>> nc.chordKind
'none'

>>> nc.chordKindStr
'N.C.'

Other values may not export:

>>> nc.chordKindStr = ''
>>> nc.write()
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
     In part (None), measure (1): NoChord object's chordKindStr must be non-empty

>>> nc.chordKind = None
>>> nc.write()
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException:
     In part (None), measure (1): NoChord object's chordKind must be 'none'

To realize the NoChord as a rest:

>>> nc2 = harmony.NoChord()
>>> nc2.writeAsChord = True
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxNote = MEX.noChordToXml(nc2)
>>> MEX.dump(mxNote)
<note>
    <rest />
    <duration>10080</duration>
    <type>quarter</type>
</note>
Tr   r   r  	root-stepCr&   r=   kindrr  z)NoChord object's chordKind must be 'none'rN   z/NoChord object's chordKindStr must be non-empty)r  r   r  r   rk  r   r  rd  rX  r   r&   r:  	chordKindr,   rR   chordKindStrsetOffsetOptionalr  r  r   )ry   csr  r  mxRootmxStepmxKindcKinds           r9   r`  MeasureExporter.noChordToXml  s#   R ??d"		2;;/A>>!$$I&	y%I*9)Iv.FK0

62Iv.<<6!)*UVV%j??j()*[\\

62??+r-)(I&r;   c                   UR                   SL a  U R                  U5      $ [        R                  " U5      nUR                  Ul        U R                  USS9n[        R                  (       a  [        U[        5      (       d   eUR                  S5      nUb  UR                  U5        UR                  S5      nUb  UR                  U5        [        S5      n[        US5      nUR                  SUR                  5        [        UR                   5      Ul        UR$                  (       aB  [        US	5      n[        UR$                  R&                  5      Ul        UR                  S
S5        UR(                  (       GaB  [        US5      n	UR*                  (       d  U	R                  SS5        [        U	S5      n
[        UR(                  R,                  5      U
l        [        U	S5      nSnUR(                  R.                  S:X  a  SnOUR                   S;  a  SnOUR                   S:X  aH  UR1                  5       R2                  UR(                  R4                  R2                  -
  S-  S:X  a  SnOJSnOGUR1                  5       R2                  UR(                  R4                  R2                  -
  S-  S:X  a  SnOSnXl        UR7                  SU5        U R8                  R;                  U5        U$ )u$
  
Convert a RomanNumeral object to either a chord (if .writeAsChord is True)
or a Harmony XML Element.

>>> rnI = roman.RomanNumeral('I', 'C')
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> MEX.currentDivisions = 10

>>> listMxChords = MEX.romanNumeralToXml(rnI)
>>> len(listMxChords)
3
>>> MEX.dump(listMxChords[1])
<note>
  <chord />
  <pitch>
    <step>E</step>
    <octave>4</octave>
  </pitch>
  <duration>10</duration>
  <type>quarter</type>
</note>

If writeAsChord is False, we create a MusicXML 4.0 <numeral> tag.
This does not work in the current version of MuseScore (which only
supports MusicXML 3.1) but outputs decently well in Finale.

>>> rnI.writeAsChord = False
>>> mxHarmonyFromRN = MEX.romanNumeralToXml(rnI)
>>> MEX.dump(mxHarmonyFromRN)
<harmony>
  <numeral>
    <numeral-root text="I">1</numeral-root>
    <numeral-key print-object="no">
      <numeral-fifths>0</numeral-fifths>
      <numeral-mode>major</numeral-mode>
    </numeral-key>
  </numeral>
  <kind>major</kind>
</harmony>

>>> complexRn = roman.RomanNumeral('#iio65', 'e-')
>>> complexRn.followsKeyChange = True
>>> complexRn.writeAsChord = False
>>> mxHarmonyFromRN = MEX.romanNumeralToXml(complexRn)
>>> MEX.dump(mxHarmonyFromRN)
<harmony>
  <numeral>
    <numeral-root text="#iio65">2</numeral-root>
    <numeral-alter location="left">1.0</numeral-alter>
    <numeral-key>
      <numeral-fifths>-6</numeral-fifths>
      <numeral-mode>minor</numeral-mode>
    </numeral-key>
  </numeral>
  <kind>diminished-seventh</kind>
  <inversion>1</inversion>
</harmony>

>>> maj6 = roman.RomanNumeral('VI7', 'd')
>>> maj6.writeAsChord = False
>>> mxHarmonyFromRN = MEX.romanNumeralToXml(maj6)
>>> MEX.dump(mxHarmonyFromRN)
<harmony>
  <numeral>
    <numeral-root text="VI7">6</numeral-root>
    <numeral-key print-object="no">
      <numeral-fifths>-1</numeral-fifths>
      <numeral-mode>natural minor</numeral-mode>
    </numeral-key>
  </numeral>
  <kind>major-seventh</kind>
</harmony>

>>> min6 = roman.RomanNumeral('vi', 'd')
>>> min6.writeAsChord = False
>>> mxHarmonyFromRN = MEX.romanNumeralToXml(min6)
>>> mxHarmonyFromRN.find('.//numeral-mode').text
'melodic minor'

>>> dim7 = roman.RomanNumeral('viiø65', 'd')
>>> dim7.writeAsChord = False
>>> mxHarmonyFromRN = MEX.romanNumeralToXml(dim7)
>>> mxHarmonyFromRN.find('.//numeral-mode').text
'harmonic minor'
>>> mxHarmonyFromRN.find('kind').text
'half-diminished'

>>> maj7 = roman.RomanNumeral('VII64', 'd')
>>> maj7.writeAsChord = False
>>> mxHarmonyFromRN = MEX.romanNumeralToXml(maj7)
>>> mxHarmonyFromRN.find('.//numeral-mode').text
'natural minor'
TFr   r  bassnumeralznumeral-rootr&   znumeral-alterlocationr  znumeral-keyra  rb  znumeral-fifthsznumeral-moder=   major)      minorr        znatural minorzmelodic minor
   zharmonic minorr   )r  rh  r   chordSymbolFromChordr  rd  r  r  r   r   r  r  r   r:  primaryFigurerR   scaleDegreer&   frontAlterationAccidentalr  r   followsKeyChangesharpsmoder  
pitchClasstonicr   r  r   )ry   rnr  r  r  mxBass	mxNumeralmxNumeralRootmxNumeralAltermxNumeralKeymxNumeralFifthsmxNumeralModemodeTexts                r9   rf  !MeasureExporter.romanNumeralToXmlg  sZ   | ??d"??2&&
 ))"-II	))"U);	??i1111'V$'V$ I&	"9n=&""2"23 0'''	?CN"%b&B&B&H&H"INz62666%i?L&&  6(7GHO#&rvv}}#5O &|^DMHvv{{g%"v-"1$GGI((266<<+B+BBbHAM.H.HGGI((266<<+B+BBbHBN.H/H!)I&I&r;   Tr  c                  UR                   SL a  U R                  U5      $ [        S5      n[        X15        U R	                  X15        U R                  X15        UR                  5       nUR                  b(  [        US5      nUR                  R                  Ul        OUb  [        US5      n[        US5      n[        UR                  5      Ul        UR                  bD  [        US5      n[        [        R                   " UR                  R"                  5      5      Ul        O[$        R'                  S/5        U$ [        US5      n	UR(                  =(       d    S	n
[*        R,                   H  n[*        R,                  U   U
:X  d  M  Un
M      [        U
5      U	l        UR.                  S
;  a  U	R1                  SUR.                  5        UR3                  5       nUS;  a  [        US5      n[        U5      Ul        UR5                  SS9=(       d    UR5                  SS9nUb  Ub  UR6                  UR6                  :w  a  [        US5      n[        US5      n[        UR                  5      Ul        UR                  bD  [        US5      n[        [        R                   " UR                  R"                  5      5      Ul        UR9                  5       nU H  n[        US5      n[        US5      n[        UR:                  5      Ul        [        US5      nUR<                  (       a)  [        UR<                  R>                  R@                  5      OSUl        [        US5      n[        URB                  5      Ul        M     U RE                  X5        U RG                  X15        U(       a  U RH                  RK                  U5        U$ )a
  
Convert a ChordSymbol object to either a chord (if .writeAsChord is True)
or a Harmony XML Element.

>>> cs = harmony.ChordSymbol()
>>> cs.root('E-')
>>> cs.bass('B-', allow_add=True)
>>> cs.inversion(2, transposeOnSet=False)
>>> cs.chordKind = 'major'
>>> cs.chordKindStr = 'M'
>>> cs
<music21.harmony.ChordSymbol E-/B->

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> MEX.currentDivisions = 10

>>> mxHarmony = MEX.chordSymbolToXml(cs)
>>> MEX.dump(mxHarmony)
<harmony>
  <root>
    <root-step>E</root-step>
    <root-alter>-1</root-alter>
  </root>
  <kind text="M">major</kind>
  <inversion>2</inversion>
  <bass>
    <bass-step>B</bass-step>
    <bass-alter>-1</bass-alter>
  </bass>
</harmony>

Now give function:

>>> cs.romanNumeral = 'I64'
>>> mxHarmony = MEX.chordSymbolToXml(cs)
>>> MEX.dump(mxHarmony)
<harmony>
  <function>I64</function>
  <kind text="M">major</kind>
  <inversion>2</inversion>
  <bass>
    <bass-step>B</bass-step>
    <bass-alter>-1</bass-alter>
  </bass>
</harmony>

>>> hd = harmony.ChordStepModification()
>>> hd.modType = 'alter'
>>> hd.interval = -1
>>> hd.degree = 3
>>> cs.addChordStepModification(hd)

>>> mxHarmony = MEX.chordSymbolToXml(cs)
>>> MEX.dump(mxHarmony)
<harmony>
  <function>I64</function>
  <kind text="M">major</kind>
  <inversion>2</inversion>
  <bass>
    <bass-step>B</bass-step>
    <bass-alter>-1</bass-alter>
  </bass>
  <degree>
    <degree-value>3</degree-value>
    <degree-alter>-1</degree-alter>
    <degree-type>alter</degree-type>
  </degree>
</harmony>

Test altered chords:

>>> f = harmony.ChordSymbol('F sus add 9')
>>> f
<music21.harmony.ChordSymbol F sus add 9>
>>> mxHarmony = MEX.chordSymbolToXml(f)
>>> MEX.dump(mxHarmony)
<harmony>
  <root>
    <root-step>F</root-step>
  </root>
  <kind>suspended-fourth</kind>
  <degree>
    <degree-value>9</degree-value>
    <degree-alter>0</degree-alter>
    <degree-type>add</degree-type>
  </degree>
</harmony>

MusicXML uses "dominant" for "dominant-seventh" so check aliases back...

>>> dom7 = harmony.ChordSymbol('C7')
>>> dom7.chordKind
'dominant-seventh'
>>> mxHarmony = MEX.chordSymbolToXml(dom7)
>>> MEX.dump(mxHarmony)
<harmony>
  <root>
    <root-step>C</root-step>
  </root>
  <kind>dominant</kind>
</harmony>

set writeAsChord to not get a symbol, but the notes.  Will return a list of notes.

>>> dom7.writeAsChord = True
>>> harmonyList = MEX.chordSymbolToXml(dom7)
>>> len(harmonyList)
4
>>> MEX.dump(harmonyList[0])
<note>
  <pitch>
    <step>C</step>
    <octave>3</octave>
  </pitch>
  <duration>10</duration>
  <type>quarter</type>
</note>
Tr   functionr  r  z
root-alterz&need either a root or a _roman to showr  rr  rN   r&   )Nr   	inversionF)r  r  z	bass-stepz
bass-alterdegreezdegree-valuezdegree-alter0zdegree-type)&r  rh  r   r  rd  rX  r  _romanr   romanNumeralfigurer&   rR   r  r  r   r  r  r  r  r  r   CHORD_ALIASESr  r:  r  r  r   getChordStepModificationsr  interval	chromaticdirectedmodTyper  r  r  r   )ry   r  r   r  csRoot
mxFunctionr  r  r  r  r  xmlAliascsInvmxInversioncsBassr  csmhdmxDegreemxDegreeValuemxDegreeAltermxDegreeTypes                         r9   rd   MeasureExporter.chordSymbolToXml  s   z ??d"??2&&I&	y%I* 	9)99 #Iz:J oo44JO	62F4Ffkk*FK   ,$V\:"6#9#9&:K:K:Q:Q#RS
 ##%M$NOIv.&--H$$X.%7  . %j??*,JJvr/ 	!$Y<K"5zK e$:T(:6>V[[FKK5O	62F4Ffkk*FK   ,$V\:"6#9#9&:K:K:Q:Q#RS
 **,B!)X6H 'x@M!$RYYM&x@MHJR[[%:%:%C%C!DY\M &h>L #BJJL " 	r-)( LL	*r;   setSoundc               $   UR                   U R                  :X  a  gUR                   U R                  -
  n[        X@R                  -  5      nUb  [	        US5      nO[        S5      n[        U5      Ul        U(       a  UR                  SS5        U$ )z
If this object has an offset different from self.offsetInMeasure,
then create and return an offset Element.

If mxObj is not None then the offset element will be appended to it.
Nr  soundr  )	r  r  r7  r  r   r   rR   r&   r:  )ry   m21ObjmxObjr  offsetDifferenceInQloffsetDifferenceInDivisionsmxOffsets          r9   r  !MeasureExporter.setOffsetOptional  s     ==D000%}}t/C/CC&)*>AVAV*V&W#!%2Hx(H78LL%(r;   c                   [        S5      n[        US5      nUR                  U5        UbJ  [        US5      (       a)  UR                  b  UR                  SUR                  5        U R                  X$US9  U$ )z
Places the mxObj <element> inside <direction><direction-type>
and sets <offset> if needed.

* Changed in v8: added `setSound` keyword (see :meth:`setOffsetOptional`).
r  r  r  r  )r   r   r   rt  r  r:  r  )ry   r
  r	  r  r  r  s         r9   placeInDirection MeasureExporter.placeInDirection  st     k*$[2BCu%v{++0@0@0LV-=-=>""6"Jr;   c                
   [        S5      n[        X!5        UR                  [        R                  ;   a  [        X!R                  5      nO&[        US5      n[        UR                  5      Ul        U R                  X!5        U R                  X!5      nU R                  XA5        UR                  nUb5  [        US5      n[        US-  5      nUR                  S[        U5      5        U R                  R                  U5        U$ )ar  
return a nested tag:
<direction><direction-type><dynamic><ff>
or whatever:

>>> ppp = dynamics.Dynamic('ppp')
>>> print('%.2f' % ppp.volumeScalar)
0.15
>>> ppp.style.relativeY = -10

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxDirection = MEX.dynamicToXml(ppp)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <dynamics default-x="-36" default-y="-80" relative-y="-10">
      <ppp />
    </dynamics>
  </direction-type>
  <sound dynamics="19" />
</direction>

appends to score.

Now with offset not zero.

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> ppp.offset = 1.0
>>> mxDirection = MEX.dynamicToXml(ppp)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <dynamics default-x="-36" default-y="-80" relative-y="-10">
      <ppp />
    </dynamics>
  </direction-type>
  <offset sound="yes">10080</offset>
  <sound dynamics="19" />
</direction>

r   zother-dynamicsr     )r   r  r8   r*   DYNAMIC_MARKSr   rR   r&   rO  r  r  volumeScalarr7  r:  r  r   )ry   r   
mxDynamicsmxThisDynamicr  vSmxSounddynamicVolumes           r9   rl  MeasureExporter.dynamicToXml  s    V Z(
z%77j...&z77;M&z3CDM!$QWWM 	
.
 ++J:+)
 ^^> g6G  SMMKK
C$67K(r;   c                    [        S5      n[        X!5        U R                  X!5        U R                  X!5      nU R                  R                  U5        U$ )aJ  
returns a segno inside a direction-type inside a direction.

appends to score

>>> s = repeat.Segno()
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxSegnoDir = MEX.segnoToXml(s)
>>> MEX.dump(mxSegnoDir)
<direction>
  <direction-type>
    <segno default-y="20" />
  </direction-type>
</direction>

>>> s.id = 'segno0'
>>> s.style.alignHorizontal = 'left'
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxSegnoDir = MEX.segnoToXml(s)
>>> MEX.dump(mxSegnoDir)
<direction>
  <direction-type>
    <segno default-y="20" halign="left" id="segno0" />
  </direction-type>
</direction>

segno)r   r  rO  r  r  r   )ry   r  mxSegnor  s       r9   rn  MeasureExporter.segnoToXml[  sL    8 '"w&/++G;K(r;   c                2   UR                   (       aU  [        S5      n[        X!5        U R                  X!5        U R	                  X!5      nU R
                  R                  U5        U$ UR                  5       nUR                  Ul        U R                  U5      $ )a>  
returns a coda inside a direction-type inside a direction IF coda.useSymbol is
True; otherwise returns a TextExpression and appends to score:

>>> c = repeat.Coda()
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxCodaDir = MEX.codaToXml(c)
>>> MEX.dump(mxCodaDir)
<direction>
  <direction-type>
    <coda default-y="20" />
  </direction-type>
</direction>

turn coda.useSymbol to `False` to get a text expression instead,
and set enclosure to NO_ENCLOSURE to explicitly generate
enclosure="none".

>>> c.useSymbol = False
>>> c.style.enclosure = style.Enclosure.NO_ENCLOSURE
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxCodaText = MEX.codaToXml(c)
>>> MEX.dump(mxCodaText)
<direction>
  <direction-type>
    <words default-y="20" enclosure="none"
        justify="center">Coda</words>
  </direction-type>
</direction>
coda)
	useSymbolr   r  rO  r  r  r   getTextExpressionr  rv  )ry   r!  mxCodar  codaTes        r9   rp  MeasureExporter.codaToXml~  s|    > >>V_F6(##F1//=KLL,++-F KKFM++F33r;   c                   / n/ nSn/ nSn[        U[        R                  5      (       a2  UR                  5       nUR                  Ul        U R                  U5      $ [        U[        R                  5      (       a  UR                  (       d  UR                  c  SnOGUR                  UR                  5        UR                  UR                  5        UR                  S5        UR                  5       =(       d    SnO[        U[        R                  5      (       a  SnUR                  UR                  4 HJ  nUR                  S5        UR                  UR                  5        UR                  UR                  5        ML     UR                  R                  5       =(       d    Sn[        S5      n	[!        X5        [#        U5       H  u  p[        S5      n[%        UR&                  5      Ul        U	R                  U5        [+        UR,                  5       H  nU	R                  [        S5      5        M     U(       d  Mz  XZ   (       a  M  [/        U	S5      n[1        [2        R4                  " US   5      5      Ul        M     [7        US	5      (       a$  UR8                  (       a  U	R;                  S	S
5        OU	R;                  S	S5        U(       a  U R=                  X5      nO)[        S5      n[!        UU5        U R=                  UU5      nU(       a;  [/        US5      nUR;                  S[1        [2        R4                  " U5      5      5        Ub  U R>                  R                  U5        [        U[        R                  5      (       aA  UR                  SS9b1  UR                  SS9nUR                  Ul        U R                  U5      nU$ )ai
  
returns a <direction> tag for a single tempo indication.

note that **two** direction tags may be added to xmlroot, the second one
as a textExpression, but only the first will be returned.

>>> mm = tempo.MetronomeMark('slow', 40, note.Note(type='half'))
>>> mm.style.justify = 'left'

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxDirection = MEX.tempoIndicationToXml(mm)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <metronome parentheses="no">
      <beat-unit>half</beat-unit>
      <per-minute>40</per-minute>
    </metronome>
  </direction-type>
  <sound tempo="80" />
</direction>

In this case, two directions were added to xmlRoot.  Here is the other one:

>>> MEX.dump(MEX.xmlRoot.findall('direction')[1])
<direction>
  <direction-type>
    <words default-y="45" font-weight="bold"
        justify="left">slow</words>
  </direction-type>
</direction>


>>> mm = tempo.MetronomeMark('slow', 40, duration.Duration(quarterLength=1.5))
>>> mxDirection = MEX.tempoIndicationToXml(mm)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <metronome parentheses="no">
      <beat-unit>quarter</beat-unit>
      <beat-unit-dot />
      <per-minute>40</per-minute>
    </metronome>
  </direction-type>
  <sound tempo="60" />
</direction>

>>> mmod1 = tempo.MetricModulation()
>>> mmod1.oldReferent = 0.75  # quarterLength
>>> mmod1.newReferent = 'quarter'  # type
>>> mxDirection = MEX.tempoIndicationToXml(mmod1)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <metronome parentheses="no">
      <beat-unit>eighth</beat-unit>
      <beat-unit-dot />
      <beat-unit>quarter</beat-unit>
    </metronome>
  </direction-type>
</direction>

>>> mmod1.newReferent = 'longa'  # music21 type w/ different musicxml name
>>> mxDirection = MEX.tempoIndicationToXml(mmod1)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <metronome parentheses="no">
      <beat-unit>eighth</beat-unit>
      <beat-unit-dot />
      <beat-unit>long</beat-unit>
    </metronome>
  </direction-type>
</direction>

This is the case where only a sound tag is added and no metronome mark

>>> mm = tempo.MetronomeMark()
>>> mm.numberSounding = 60

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxDirection = MEX.tempoIndicationToXml(mm)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <words />
  </direction-type>
  <sound tempo="60" />
</direction>

This is the case of a TempoText.

>>> tt = tempo.TempoText('Andante')
>>> mxDirection = MEX.tempoIndicationToXml(tt)
>>> MEX.dump(mxDirection)
<direction>
  <direction-type>
    <words default-y="45" font-weight="bold">Andante</words>
  </direction-type>
</direction>

Fr   T	metronomez	beat-unitzbeat-unit-dotz
per-minuter   r  r  rb  wordsr  r%   )returnImplicit) r   r%   rq  r#  r  rv  rs  numberImplicitr   r   referentgetQuarterBPMrt  oldMetronomenewMetronomer   r  rC  r:   r  r&   r   r  r   rR   r   r  rt  r  r:  r  r  )ry   tidursnumbershideNumericalMetro
hideNumbersoundingQuarterBPMtesubmxMetror   r   mxSubr  mxPerMinuter  mxWordsr  unused_mxDirectionTexts                      r9   rr  $MeasureExporter.tempoIndicationToXml  s   V "
  b%//**%%'B		BI++B//b%--..  BII$5%)"BKK(ryy)!!%( "$!1!1!3!:sE2233 "'9!!$'CLL)szz* :
 "$!>!>!@!GC +&w#dODAK(E+AFF3EJNN5!%*166]!w78 &3wz}}(,?#&v'='=gaj'I#J  $ 2}%%"..KKu-KKt, //<Kg&G7B'//<K g6GKKV%;%;<N%O!PQ)LL,b%--..##5#9E)))?II	)-)A)A")E&r;   c                    [        S5      nU R                  X!5        [        UR                  5      Ul        U R                  X!5      nU R                  X1S5        U R                  R                  U5        U$ )a  
Convert a RehearsalMark object to a MusicXML <direction> tag with a <rehearsal> tag
inside it.

>>> rm = expressions.RehearsalMark('II')
>>> rm.style.enclosure = 'square'
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxRehearsal = MEX.rehearsalMarkToXml(rm)
>>> MEX.dump(mxRehearsal)
<direction>
  <direction-type>
    <rehearsal enclosure="square" halign="center" valign="middle">II</rehearsal>
    </direction-type>
  </direction>
	rehearsalr  )	r   rQ  rR   rT  r&   r  rC  r  r   )ry   rmmxRehearsalr  s       r9   ry  "MeasureExporter.rehearsalMarkToXmlg  sd    " k*{/rzz?++K<=K(r;   c                X   SnUR                  5       nU H&  n[        U[        R                  5      (       d  M$  Un  O   Uc  g/ n[        U[        R                  5      (       Ga  UR
                  [        R                  R                  [        R                  R                  4;   a#  [        S5      /nUS   R                  SS5        GO2UR
                  [        R                  R                  :X  a  [        S5      /nUR                  [        R                  R                  :X  a  US   R                  SS5        GOUR                  [        R                  R                  :X  a  US   R                  SS5        GOUS   R                  SS5        GOiUR
                  [        R                  R                   :X  a  [        S5      [        S5      /nUS   R                  SS	5        UR                  [        R                  R                  :X  a  US
   R                  SS5        OUR                  [        R                  R                  :X  a  US
   R                  SS5        OUS
   R                  SS5        Og[        U[        R"                  5      (       a"  [        S5      /nUS   R                  SS5        OB[        U[        R$                  5      (       a"  [        S5      /nUS   R                  SS5        OgU H  nUR
                  [        R                  R                  [        R                  R                  4;   a  UR                  SS5        OUR                  SS5        U R'                  Xa5      nU R)                  XqS5        U R*                  R-                  U5        M     W$ )a  
Convert a PedalObject object (i.e. PedalBounce, PedalGapStart,
PedalGapEnds) to a MusicXML <direction> tag with a <pedal> tag
inside it.  Attributes like pedalForm are stored in the enclosing
PedalMark spanner.

>>> pm = expressions.PedalMark()
>>> po = expressions.PedalBounce()
>>> pm.addSpannedElements(po)
>>> pm.pedalForm = expressions.PedalForm.Line
>>> pm.pedalType = expressions.PedalType.Sustain
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxPedal = MEX.pedalObjectToXml(po)
>>> MEX.dump(mxPedal)
<direction>
  <direction-type>
    <pedal line="yes" type="change" />
  </direction-type>
</direction>

>>> pm = expressions.PedalMark()
>>> po = expressions.PedalGapStart()
>>> pm.addSpannedElements(po)
>>> pm.pedalForm = expressions.PedalForm.Line
>>> pm.pedalType = expressions.PedalType.Sustain
>>> po.placement = 'above'
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxPedal = MEX.pedalObjectToXml(po)
>>> MEX.dump(mxPedal)
<direction placement="above">
  <direction-type>
    <pedal line="yes" type="discontinue" />
  </direction-type>
</direction>

>>> pm = expressions.PedalMark()
>>> po = expressions.PedalGapEnd()
>>> pm.addSpannedElements(po)
>>> pm.pedalForm = expressions.PedalForm.Line
>>> pm.pedalType = expressions.PedalType.Sustain
>>> po.placement = 'below'
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxPedal = MEX.pedalObjectToXml(po)
>>> MEX.dump(mxPedal)
<direction placement="below">
  <direction-type>
    <pedal line="yes" type="resume" />
  </direction-type>
</direction>
Nr  r   r  changer  r  sustainr  r   discontinueresumer  r  r  r  )getSpannerSitesr   r   r  PedalBouncer  r   r  r  r   r:  	SymbolAltr  r  Sustainr  SymbolPedalGapStartPedalGapEndr  rC  r  r   )ry   popmspannersr#  mxPedalsmxPedalr  s           r9   r{   MeasureExporter.pedalObjectToXml  s   h *.*,*<*<*>B"k3344 
 : "$b+1122|| 5 5 : :K<Q<Q<\<\]] $G,-1!6!6!@!@@#G,-<<;#8#8#@#@@QKOOFG4\\[%:%:%D%DDQKOOFK8 QKOOFI6!6!6!=!==#G,gg.>?/<<;#8#8#@#@@QKOOFG4\\[%:%:%D%DDQKOOFK8 QKOOFI6 K5566()HQKOOFM2K3344()HQKOOFH-G|| 5 5 : :K<Q<Q<\<\]]FE*FE* //<K##K[ALL,   r;   c                    [        S5      nUR                  SS5        UR                  SS5        U R                  X!5      nU R                  X1S5        U$ )Nr  r  rG  r  r  r  )r   r:  r  rC  )ry   rP  rS  r  s       r9   r  &MeasureExporter.makePedalResumeLineXml  sN    '"FH%FE"++G8=r;   c                   [        S5      n[        U[        R                  5      (       a  Un[	        UR
                  5      Ul        Of[        U[        R                  5      (       a<  UR                  5       nUR                  Ul
        [	        UR
                  5      Ul        O[        S5      eU R                  X#5        U R                  X#SS9nU R                  R                  U5        U$ )z|
Convert a TextExpression or RepeatExpression to a MusicXML mxDirection type.
returns a musicxml.mxObjects.Direction object
r)  z3teOrRe must be a TextExpression or RepeatExpressionFr  )r   r   r   ru  rR   rT  r&   r    rw  r#  r  r,   rQ  r  r  r   )ry   teOrRer;  r6  r  s        r9   rv  #MeasureExporter.textExpressionToXml	  s     '"fk8899Brzz?GL 7 788))+BBIrzz?GL)*_``w+++G%+HK(r;   c                    U R                   S:X  a  g[        S5      nU" U5      nUR                  U5        U R                  R                  U5        U$ )aY  
given a Clef, KeySignature, or TimeSignature which is in .elements and not m.clef,
etc. insert it in self.xmlRoot as
part of the current mxAttributes using methodToMx as a wrapper function.

(or insert into a new mxAttributes if Clef/KeySignature/etc. is not at the beginning
of the measure and not at the same point as an existing mxAttributes)

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> MEX.offsetInMeasure = 3.0
>>> cl = clef.BassClef()
>>> methodToMx = MEX.clefToXml
>>> mxAttributes = MEX.wrapObjectInAttributes(cl, methodToMx)
>>> MEX.dump(mxAttributes)
<attributes>
  <clef>
    <sign>F</sign>
    <line>4</line>
  </clef>
</attributes>

Also puts it in MEX.xmlRoot.

If offsetInMeasure is 0.0 then nothing is done or returned:

>>> MEX.offsetInMeasure = 0.0
>>> nothing = MEX.wrapObjectInAttributes(cl, methodToMx)
>>> nothing is None
True
r   N
attributes)r  r   r   r  )ry   objectToWrap
methodToMxmxAttributesr
  s        r9   r  &MeasureExporter.wrapObjectInAttributes$  sP    H 3&|,<(E"L)r;   c                   U R                   c  gU R                   R                  c  gUR                  c  gU R                   R                  R                  UR                  5      nUc  g[	        X5      nU" [	        XB5      5      $ )a  
If this measure is part of a subsequent PartStaff in a joinable staff
group (e.g. the left hand of a keyboard part), then look up the
corresponding measure by number in the previous PartStaff, retrieve
the `attr` value there, and compare it to `obj` by `comparison`.
F)r   r  r  r   rP   )ry   rz   attr
comparisonmaybe_measurecomparison_wrappers         r9    _matchesPreviousPartStaffInGroup0MeasureExporter._matchesPreviousPartStaffInGroupT  sy     ;;;;//7
 $<<DDSEVEVW $S5!'-">??r;   c           	     n   [        S5      nUR                  (       d  [        XSS5        [        XSSSS9  O[        R                  (       a  UR
                  c   e[        UR
                  5       Hd  u  p4UR                  S:X  a  M  US:  a.  [        US5      nUR                  (       a  UR                  Ul
        [        XBSS5        [        XBSSSS9  Mf     UR                  b%  UR                  S	[        UR                  5      5        UR                  b&  UR                  S
[        UR                  5      5        O2UR                  b%  UR                  S
[        UR                  5      5        U R                  X!SS5        U R!                  X!5        U R#                  X!5        U R%                  X!5        U$ )z
Translate a music21 :class:`~music21.note.Lyric` object
to a <lyric> tag.

Lyrics have attribute list %justify, %position, %placement, %color, %print-object
lyricsyllabicr&   T)rL   	compositer   elisionr   r   )rF  r  )r   isCompositerX   r  r  
componentsrC  ri  r   elisionBeforer&   
identifierr:  rR   r   rC  rd  r^  r\  )ry   lyr  r   	component	mxElisions         r9   r  MeasureExporter.lyricToXmls  s[    '"~~$R*jI$R&&TR }}000 )"-- 8%%46 *7I >I ..)2)@)@	(ZT(VVX\] !9" ==$KKBMM 2399 KK#bii.1]]&KK#bmm"45 8 8	: 	G(g"%r;   c                p    / nUR                    H#  nUR                  U R                  U5      5        M%     U$ )aN  
Returns a list of <beam> tags
from a :class:`~music21.beam.Beams` object


>>> a = beam.Beams()
>>> a.fill(2, type='start')

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxBeamList = MEX.beamsToXml(a)
>>> len(mxBeamList)
2
>>> for b in mxBeamList:
...     MEX.dump(b)
<beam number="1">begin</beam>
<beam number="2">begin</beam>
)	beamsListr   	beamToXml)ry   r3  
mxBeamListbeamObjs       r9   r  MeasureExporter.beamsToXml  s4    & 
GdnnW56 'r;   c                   [        S5      n[        X!5        SSSS.nUR                  U;   a  X1R                     Ul        OqUR                  S:X  aH  UR                  S:X  a  SUl        OIUR                  S	:X  a  S
Ul        O1[        SUR                  -  5      e[        SUR                   S35      eUR                  S[        UR                  5      5        U R                  X!5        U R                  X!SS5        U$ )ah  
Returns an ElementTree Element from a :class:`~music21.beam.Beam` object

>>> a = beam.Beam()
>>> a.type = 'start'
>>> a.number = 1

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> b = MEX.beamToXml(a)
>>> b
<Element 'beam' at 0x104f3a728>
>>> MEX.dump(b)
<beam number="1">begin</beam>

>>> a.type = 'continue'
>>> b = MEX.beamToXml(a)
>>> MEX.dump(b)
<beam number="1">continue</beam>

>>> a.type = 'stop'
>>> b = MEX.beamToXml(a)
>>> MEX.dump(b)
<beam number="1">end</beam>

>>> a.type = 'partial'
>>> a.direction = 'left'
>>> b = MEX.beamToXml(a)
>>> MEX.dump(b)
<beam number="1">backward hook</beam>

>>> a.direction = 'right'
>>> a.id = 'beam1'
>>> b = MEX.beamToXml(a)
>>> MEX.dump(b)
<beam id="beam1" number="1">forward hook</beam>

>>> a.direction = None
>>> b = MEX.beamToXml(a)
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException: partial beam defined
    without a proper direction set (set to None)

>>> a.type = 'crazy'
>>> b = MEX.beamToXml(a)
Traceback (most recent call last):
music21.musicxml.xmlObjects.MusicXMLExportException: unexpected beam type
    encountered (crazy)
beambeginrC  end)r  rC  r  partialr  zbackward hookr  zforward hookz?partial beam defined without a proper direction set (set to %s)z"unexpected beam type encountered ()r   fan)r   r  r  r&   r  r,   r:  rR   r   r^  rC  )ry   
beamObjectmxBeam
beamToTypes       r9   rv  MeasureExporter.beamToXml  s    b v*&",#
 ??j($__5FK__	)##v--%%0,-U(()* * *4Z__4EQG  	

8S!2!234
 	f) 	E5Ar;   c                    U R                   n[        US5      (       d  gU R                  nU R                   R                  nUc!  U(       a  US   R	                  U5      (       d  gU R                  US5        g)zR
Calls self.setBarline for the right side if the measure has a .rightBarline set.
rightBarlineNr   r  )r"   rt  r  r  r  
setBarline)ry   r   r  r  s       r9   r  MeasureExporter.setRightBarline  se     KKq.))__
{{// #:a=+?+?+B+B OOL'2r;   c                    U R                   n[        US5      (       d  gU R                  nUR                  nUc!  U(       a  US   R	                  U5      (       d  gU R                  US5        g)zP
Calls self.setBarline for the left side if the measure has a .leftBarline set.
leftBarlineNr   r  )r"   rt  r  r  r  r  )ry   r   r  r  s       r9   r  MeasureExporter.setLeftBarline!  s_     KKq-((__
mm#:a=+@+@+C+C OOK0r;   c                   SnUc  [        S5      nOM[        U[        R                  5      (       a  [        S5      nU R	                  U5      nOU R                  U5      n[        XA5        UR                  SU5        U R                  (       Ga  US:X  a0  U R                  S   R                  U R                  5      (       a  SnO8US:X  a0  U R                  S   R                  U R                  5      (       a  SnOS	nU(       a  [        S
5      nU R                  S   R                  n[        US   5      nUS:X  a  S	nUSS  H  n	US[        U	5      -   -  nM     UR                  SU5        UR                  SU5        UR                  U5        Ub  UR                  U5        U R                  R                  U5        U$ )zX
sets either a left or right barline from a
bar.Barline() object or bar.Repeat() object
Nbarliner  r  r   r  r  r  r=   endingr  r   rs  r   r  )r   r   r
   RepeatrepeatToXmlbarlineToXmlr  r:  r  r  r"   r  numberRangerR   r   r  )
ry   r  positionmxRepeat	mxBarline
endingTypemxEnding
numberList	numberStrnums
             r9   r  MeasureExporter.setBarline2  s   
 ?	*I'3::..#I.	++G4 --g6	y*j(+ ???6!dooa&8&@&@&M&M$
W$);)B)B4;;)O)O#

"8,!__Q/;;

1.	# "I%ab>Cs3x/I *Xy1VZ0  *X& 	I&r;   c                    [        S5      n[        US5      nUR                  5       Ul        UR                  b  UR                  SUR                  5        U$ )a  
Translate a music21 bar.Bar object to an mxBar
while making two substitutions: double -> light-light
and final -> light-heavy as shown below.

>>> b = bar.Barline('final')
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxBarline = MEX.barlineToXml(b)
>>> MEX.dump(mxBarline)
<barline>
  <bar-style>light-heavy</bar-style>
</barline>

>>> b.location = 'right'
>>> mxBarline = MEX.barlineToXml(b)
>>> MEX.dump(mxBarline)
<barline location="right">
  <bar-style>light-heavy</bar-style>
</barline>
r  z	bar-styler  )r   r   musicXMLBarStyler&   r  r:  )ry   	barObjectr  
mxBarStyles       r9   r  MeasureExporter.barlineToXmli  sP    , I&		;7
#446
)MM*i&8&89r;   c                N   [        S5      nUR                  S:X  a  UR                  SS5        ODUR                  S:X  a  UR                  SS5        O![        R                  " SUR                  5      eUR
                  b%  UR                  S[        UR
                  5      5        U$ )	aO  
returns a <repeat> tag from a barline object.

>>> b = bar.Repeat(direction='end')
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxRepeat = MEX.repeatToXml(b)
>>> MEX.dump(mxRepeat)
<repeat direction="backward" />

>>> b.times = 3
>>> mxRepeat = MEX.repeatToXml(b)
>>> MEX.dump(mxRepeat)
<repeat direction="backward" times="3" />
r    r  r  r  r}  backwardzcannot handle direction format:times)r   r  r:  r
   BarExceptionr  rR   )ry   r  r  s      r9   r  MeasureExporter.repeatToXml  s      8$;;'!LLi0[[E!LLj1 ""#DakkRR77LL#agg,/ r;   c                t   U R                   n[        S5      n[        R                  U l        U R
                  b$  U R                  U R
                  R                  :w  aA  [        US5      n[        U R                  5      Ul	        U R                  U R
                  l        UR                  bK  U R                  UR                  S5      (       d*  UR                  U R                  UR                  5      5        UR                  bJ  U R                  UR                  SSS9(       d*  UR                  U R                  UR                  5      5        [!        UR#                  [$        R&                  5      5      nU(       a#  UR                  U R                  US   5      5        UR(                  b*  UR                  U R+                  UR(                  5      5        UR#                  [,        R.                  5      nU(       a%  US   nUR                  U R1                  U5      5        U R2                  b*  UR                  U R5                  U R2                  5      5        U R7                  5       nUb  UR                  U5        [9        U5      S:  d  UR:                  (       a  U R<                  R                  U5        U$ )z
creates an <attributes> tag at the start of the measure.

We create one for each measure unless it is identical to self.parent.mxAttributes
r[  	divisionsr-  r   
ratioEqual)rb  r   )r"   r   r   r  r  r   r   r   rR   r&   r-  re  r   r}  r   r~  r  r   r   SenzaMisuraTimeSignaturer   r|  r   StaffLayoutstaffLayoutToXmlStaffDetailsr  intervalToXmlTransposemeasureStyler  rg  r  )ry   r   r^  mxDivisionssmtsfoundslmxMeasureStyles           r9   r  6MeasureExporter.setMxAttributesObjectForStartOfMeasure  s    KK|,
 !) < <;;$"7"74;;;T;T"T$\;?K"4#8#89K(,(=(=DKK%NN&==NNN   6 6q~~ FGOO'==OO_ >   7 7 HIA(()G)GHI 7 7Q @A
 66qvv 67$$V%7%78qB A A" EF%%1 ; ;D<V<V WX **,%/|q L$7$7LL-r;   c                   U R                   nSnSnUR                  [        R                  5      nU(       a  US   R	                  S5      nU(       ay  US   nUR                  US   5      (       a[  [        S5      nUR                  (       a  UR                  SS5        OUR                  SS5        [        UR                  5      Ul        Ub  [        S5      nUR                  U5        U$ )	z
return a <measure-style> Element or None according to the contents of the Stream.

Currently, only multiple-rest is supported.
Nr   MultiMeasureRestzmultiple-restzuse-symbolsr  rb  zmeasure-style)r"   r   r   r  rH  r  r   
useSymbolsr:  rR   numRestsr&   r   )ry   r   r  mxMultipleRestrestshasMMRfirstRestMMRs          r9   r  MeasureExporter.measureStyle  s     KK$$TYY/1X--.@AF%ay''a11%,_%=N#..&**=%@&**=$?*-l.C.C*DN'%$_5N !!.1r;   c                   [        S5      nUR                  SL a  UR                  SS5        OUR                  SS5        [        R                  R
                  nUR                  UR                  UR                  4;  a'  [        US5      nUR                  R                  Ul        UR                  b&  [        US5      n[        UR                  5      Ul        U$ )a  
Convert a :class:`~music21.layout.StaffLayout` object to a
<staff-details> element.

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> sl = layout.StaffLayout()
>>> sl.staffLines = 3  # tenor drums?
>>> sl.staffType = stream.enums.StaffType.CUE
>>> sl.hidden = True
>>> mxDetails = MEX.staffLayoutToXmlStaffDetails(sl)
>>> MEX.dump(mxDetails)
<staff-details print-object="no">
      <staff-type>cue</staff-type>
      <staff-lines>3</staff-lines>
</staff-details>
zstaff-detailsTra  rb  r  z
staff-typezstaff-lines)r   rH  r:  r"   r   	StaffType	staffTypeREGULAROTHERr   r8   r&   
staffLinesrR   )ry   r  mxStaffDetailsr  mxStaffTypemxStaffLiness         r9   r  ,MeasureExporter.staffLayoutToXmlStaffDetails   s    , !1%~t4~u5LL**	  ):):IOO(LL$^\BK*44::K!!-%nmDL #K$:$: ;L
 r;   c                   [        S5      n[        X!5        [        U[        R                  5      (       a,  [        US5      nUR                  b  UR                  Ul        U$ [        S UR                  R                  5        5       5      nUR                  (       a  [        R                  R                  U5      nU H=  u  pV[        US5      n[        U5      Ul        [        US5      n[        U5      Ul        M?     UR                  (       a  UR                  SS5        O,UR                   S:w  a  UR                  SUR                   5        U R#                  X!5        U R%                  X!5        U$ )	a  
Returns a single <time> tag from a meter.TimeSignature object.

Compound meters are represented as multiple pairs of beat
and beat-type elements

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> a = meter.TimeSignature('3/4')
>>> b = MEX.timeSignatureToXml(a)
>>> MEX.dump(b)
<time>
  <beats>3</beats>
  <beat-type>4</beat-type>
</time>

>>> a = meter.TimeSignature('3/4+2/4')
>>> b = MEX.timeSignatureToXml(a)
>>> MEX.dump(b)
<time>
  <beats>3</beats>
  <beat-type>4</beat-type>
  <beats>2</beats>
  <beat-type>4</beat-type>
</time>

>>> a.setDisplay('5/4')
>>> b = MEX.timeSignatureToXml(a)
>>> MEX.dump(b)
<time>
  <beats>5</beats>
  <beat-type>4</beat-type>
</time>

>>> a = meter.TimeSignature('4/4')
>>> a.symbol = 'common'
>>> b = MEX.timeSignatureToXml(a)
>>> MEX.dump(b)
<time symbol="common">
  <beats>4</beats>
  <beat-type>4</beat-type>
</time>

>>> a.symbol = ''
>>> a.symbolizeDenominator = True
>>> b = MEX.timeSignatureToXml(a)
>>> MEX.dump(b)
<time symbol="note">
  <beats>4</beats>
  <beat-type>4</beat-type>
</time>

>>> sm = meter.SenzaMisuraTimeSignature('free')
>>> b = MEX.timeSignatureToXml(sm)
>>> MEX.dump(b)
<time>
  <senza-misura>free</senza-misura>
</time>
timezsenza-misurac              3  P   #    U  H  oR                   UR                  4v   M     g 7fr   )r  r  )r/  mts     r9   r1  5MeasureExporter.timeSignatureToXml.<locals>.<genexpr>n  s     \?[||R^^4?[s   $&beatsz	beat-typer  r   r=   )r   r  r   r   r  r   r&   tupledisplaySequenceflattensummedNumeratortoolsfractionToSlashMixedrR   symbolizeDenominatorr:  r  rO  rd  )	ry   tsmxTimemxSenzaMisurafListr   r   mxBeats
mxBeatTypes	            r9   r~  "MeasureExporter.timeSignatureToXml)  s#   x v"b%8899&v~>Mww"%'WW"M \r?Q?Q?Y?Y?[\\ KK44U;EDA 1Gq6GL#FK8J!!fJO	  ""JJx(YY"_JJx+ 	+F'r;   c                   [         n[        S5      n[        X15        U R                  X15        UR                  (       d3  U" XSS5        [        US5      (       a  UR                  b
  U" XSS5        O|UR                   Hl  nU" XCSS5        UR                  nUc  [        R                  " S5      n[        US5      n[        [        R                  " UR                  5      5      Ul        Mn     [#        UR                  5       HX  u  ptUR$                  c  M  [        US	5      n[        UR$                  5      Ul        UR'                  S
[        US-   5      5        MZ     U$ )a/  
returns a key tag from a music21
key.KeySignature or key.Key object

>>> ks = key.KeySignature(-3)
>>> ks
<music21.key.KeySignature of 3 flats>

>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxKey = MEX.keySignatureToXml(ks)
>>> MEX.dump(mxKey)
<key>
  <fifths>-3</fifths>
</key>

>>> ks.mode = 'major'
>>> mxKey = MEX.keySignatureToXml(ks)
>>> MEX.dump(mxKey)
<key>
  <fifths>-3</fifths>
  <mode>major</mode>
</key>

>>> ksNonTrad = key.KeySignature(sharps=None)
>>> ksNonTrad.alteredPitches = ['C#', 'E-4']
>>> ksNonTrad
<music21.key.KeySignature of pitches: [C#, E-4]>

>>> mxKeyNonTrad = MEX.keySignatureToXml(ksNonTrad)
>>> MEX.dump(mxKeyNonTrad)
<key>
  <key-step>C</key-step>
  <key-alter>1</key-alter>
  <key-step>E</key-step>
  <key-alter>-1</key-alter>
  <key-octave number="2">4</key-octave>
</key>
r   fifthsr  r  zkey-stepr  r   z	key-alterz
key-octaver   r   )rX   r   r  rX  isNonTraditionalrt  r  alteredPitchesr  r   
Accidentalr   rR   r   r  r  r&   rC  r  r:  )	ry   keyOrKeySignaturer  mxKeyr   r  r  r   mxKeyOctaves	            r9   r}  !MeasureExporter.keySignatureToXml  s,   P (u054 !11 "8X>(&116G6L6L6X&vv> '55Qz62LL9((+A$UK8"6#9#9!''#BC 6 />>?DAxx#(=#&qxx= #a!e*5	 @ r;   c                   [        S5      n[        X!5        U R                  X!5        UR                  nUc  [	        S5        Sn[        US5      nX4l        [        XSS5        UR                  S;  a  [        XSS5        U$ )	a  
Given a music21 Clef object, return a MusicXML clef
tag.

>>> gc = clef.GClef()
>>> gc
<music21.clef.GClef>
>>> MEX = musicxml.m21ToXml.MeasureExporter()
>>> mxc = MEX.clefToXml(gc)
>>> MEX.dump(mxc)
<clef>
  <sign>G</sign>
</clef>

>>> b = clef.Treble8vbClef()
>>> b.octaveChange
-1
>>> mxc2 = MEX.clefToXml(b)
>>> MEX.dump(mxc2)
<clef>
  <sign>G</sign>
  <line>2</line>
  <clef-octave-change>-1</clef-octave-change>
</clef>

>>> pc = clef.PercussionClef()
>>> mxc3 = MEX.clefToXml(pc)
>>> MEX.dump(mxc3)
<clef>
  <sign>percussion</sign>
</clef>

Clefs without signs get exported as G clefs with a warning

>>> generic = clef.Clef()
>>> mxc4 = MEX.clefToXml(generic)
Clef with no .sign exported; setting as a G clef
>>> MEX.dump(mxc4)
<clef>
  <sign>G</sign>
</clef>
r   z0Clef with no .sign exported; setting as a G clefGr  r  r   Nzclef-octave-changeoctaveChange)	r   r  rX  r  r  r   r&   rX   r  )ry   clefObjmxClefr  mxSigns        r9   r|  MeasureExporter.clefToXml  s    V v'6+ ||<DEDFF+ &&Ay0$W6JN[r;   c                   Uc  U R                   nUR                  R                  R                  n[	        [        U5      S-
  S5      u  p4[        UR                  R                  5      S-  nUS:  a  US-  nUS-  nUS-  n[        S5      n[        Xa5        [        US5      n[        U5      Ul        [        US5      n[        U5      Ul        US:w  a  [        US	5      n	[        U5      U	l        U$ )
aW  
>>> ME = musicxml.m21ToXml.MeasureExporter()
>>> i = interval.Interval('P5')
>>> mxTranspose = ME.intervalToXmlTranspose(i)
>>> ME.dump(mxTranspose)
<transpose>
  <diatonic>4</diatonic>
  <chromatic>7</chromatic>
</transpose>


>>> i = interval.Interval('A13')
>>> mxTranspose = ME.intervalToXmlTranspose(i)
>>> ME.dump(mxTranspose)
<transpose>
  <diatonic>5</diatonic>
  <chromatic>10</chromatic>
  <octave-change>1</octave-change>
</transpose>

>>> i = interval.Interval('-M6')
>>> mxTranspose = ME.intervalToXmlTranspose(i)
>>> ME.dump(mxTranspose)
<transpose>
  <diatonic>-5</diatonic>
  <chromatic>-9</chromatic>
</transpose>


>>> i = interval.Interval('-M9')
>>> mxTranspose = ME.intervalToXmlTranspose(i)
>>> ME.dump(mxTranspose)
<transpose>
  <diatonic>-1</diatonic>
  <chromatic>-2</chromatic>
  <octave-change>-1</octave-change>
</transpose>

r   r  r  r   r&  	transposediatonicr  zoctave-change)r  r  genericr  divmodabsr  r  r   r  r   rR   r&   )
ry   r   genericStepsmusicxmlOctaveShiftmusicxmlDiatonicmusicxmlChromaticr  
mxDiatonicmxChromaticmxOctaveChanges
             r9   r  &MeasureExporter.intervalToXmlTranspose  s    V 9**Azz))2206s<7H17La0P- 5 56;!"2%#k*{&Z8
./
 k:01!#'_EN"%&9":Nr;   c                v   U R                   nUR                  [        R                  5      nU(       d  gSnUR                  [        R                  5      nU(       a  US   nU R                  U5      nUR                  [        R                  5      nU(       a+  US   nUc  U R                  U5      nOU R                  Xc5        UR                  [        R                  5      nU(       a+  US   nUc  U R                  U5      nOU R                  Xc5        UR                  (       av  UR                  R                  b_  Uc  [        S5      n[        US5      nUR                  R                  Ul        UR                  R                   nUb  U R#                  Xx5        Ub  U R$                  R'                  U5        gg)zE
Creates a <print> element and appends it to root, if one is needed.
Nr   r  zmeasure-numbering)r"   r   r   r  
PageLayoutr  SystemLayoutr  r  staffLayoutToXmlPrintr4  r$   r   r   r   r&   measureNumberStylerO  r  r   )	ry   r   foundAnyr  r  plr  mxMeasureNumberingmnStyles	            r9   r  MeasureExporter.setMxPrint[  sy    KK ''(9(9:++F,=,=>qB//3G++F,?,?@qB55b9++B8++F,>,>?qB44R8**27   QWW%=%=%I!'*!+G5H!I&'gg&>&>#gg00G"''(:D LL( r;   c                |    Uc  [        S5      n[        X!5        U R                  U5      nUR                  U5        U$ )Nr  )r   r  r  r   )ry   r  r  r  s       r9   r  %MeasureExporter.staffLayoutToXmlPrint  s:    ?g&Gw,88E}%r;   c                2   U R                   n[        US5      (       a*  U R                  R                  SUR	                  5       5        [        XR                  SSS 5        [        US5      (       a&  UR                  b  [        XR                  SS5        ggg)	z
sets the attributes (x=y) for a measure,
that is, number, implicit, and layoutWidth

Does not create the <attributes> tag. That's elsewhere.
measureNumberWithSuffixr   implicit
showNumberc                v    [         R                  " U [        R                  R                  R
                  L 5      $ r   )r*   r8  r"   r   
ShowNumberNEVER)showNums    r9   <lambda>1MeasureExporter.setMxAttributes.<locals>.<lambda>  s$    J55gAXAXA^A^6^_r;   layoutWidthNwidth)r"   rt  r  r:  r  r  r  )ry   r   s     r9   r  MeasureExporter.setMxAttributes  s     KK1/00LLXq'@'@'BC!||Z_	a 1m$$)B%awN *C$r;   c                z    U R                   R                  U R                  5      nUR                  S5      U l        g)z.
Makes a set of spanners from repeat brackets
RepeatBracketN)r  r  r"   r  r  )ry   spannersOnStreams     r9   r  MeasureExporter.setRbSpanners  s1      --AA$++N*55oFr;   c                   U R                   c  gU R                   R                  c  gU R                  nUR                  U R                   R                  5      U l        U R                   R                  R                  U R                  U R                  UR                  R                  -   SS9nU(       d  gUR                  5       nUR                  c  gUR                  U l
        g)z
Set the transposition interval based on whether the active
instrument for this period has a transposition object.

Stores in self.transpositionInterval.  Returns None
NF)includeEndBoundary)r   r"   getOffsetBySiter  r  r   r   r   r   transpositionr  )ry   r   instSubStream
instSubObjs       r9   r  MeasureExporter.setTranspose  s     ;;;;%KK"#"3"3DKK4F4F"G44HH####ajj&>&>>$ I & "((*
##+%/%=%=" r;   )r  r  r  r  r  r  r  r  r  r   r  r  r"   r  r  rZ  )r  zstream.Measure | Noner   zPartExporter | None)r  r.   )r   zstream.Measure | stream.Voicer  r  r   r  )r  r   r[  )r  base.Music21Objectr   z/tuple[t.Sequence[Element], t.Sequence[Element]])r  rR   r#  zspanner.Spannerr   zdict[str, t.Any])r(  zexpressions.ArpeggioMarkSpannerr   r7  r  )rz   r  r<  r7  r  zspanner.SpannerBundle | Noner   list[Element]r   )
rz   r  r<  r7  r=  r  r>  zspanner.SpannerBundler   r  )r   note.GeneralNote)r   note.NotRestr  r   r  chord.Chord | None)r  z	note.Rest)r  zchord.ChordBaser   r  )r  zduration.Durationr   )rp  znote.Unpitchedr<  r7  r  zchord.ChordBase | Noner   r   r  )r   r  )r  ztablature.ChordWithFretBoardr   r   )r  r   r   r   r  r"  r   r  )r   r!  r   r   )r(  zexpressions.Ornamentr   r  )r   r   r<  r7  r  r"  r   r  )rD  ztie.Tie)r   )r|  zexpressions.Expressionr   r   )r  articulations.Articulationr   r   )r  r#  )r  r   r  zarticulations.FretBendr   r  )r  r   r  zarticulations.StringHarmonicr   r  )r  zharmony.NoChordr   r   )r  zroman.RomanNumeralr   Element | list[Element])r  zharmony.ChordSymbolr   r  r   r$  )r	  r  r
  r  r  r  r   r  )r
  r   r	  zbase.Music21Object | Noner  r  r   r   )r   zdynamics.Dynamicr   r   )r  zrepeat.Segnor   r   )r!  zrepeat.Codar   r   )r0  ztempo.TempoIndicationr   r   )r@  zexpressions.RehearsalMarkr   r   )rO  zexpressions.PedalObjectr   r  )rP  zexpressions.PedalMarkr   r   )rX  z4expressions.TextExpression | repeat.RepeatExpressionr   r   )r\  r  r]  zt.Callable[[t.Any], Element]r   r  )r-  __eq__)rz   r  ra  rR   rb  rR   r   r  )rp  z
note.Lyric)r  z4meter.TimeSignature | meter.SenzaMisuraTimeSignature)Xr   r   r   r   r   r  r  r  r{   r   r  r  r  r  r   r  r  r  r  r  r  r  r*  rH  rR  r7  r^  r  rk  rh  r~  r|  rj  r  r  rb  r  r  r  r  r*  r  r  r5  r  r  r4  r8  r3  r  r  r`  rf  rd  r  r  rl  rn  rp  rr  ry  r{  r  rv  r  re  r  r  rv  r  r  r  r  r  r  r  r  r~  r}  r|  r  r  r  r  r  r  r   r  r  s   @r9   r
  r
    s   "	
. "-	" ))4 48-1!60!6*!6 !6F(I2-6-< "'	@#&@# 	@#
 
@#J *8)>)>N  'N `*^!"^! 
9^!@ G GR ' 'R	 !":>	XX X 8	X
 
XtHT'-'- '- !	'-
 "'- 
'-RfP	%A)%A")%A (8%AN{zRh 4 01;?*)*),* %9* EL*X$L2h"""P )-	"*"* "* &	"*
 
"*H,\ !J !"(,	  &	
 
B@KZkZwrC"4C" 
C"JN,gR :7 :7x .. ..`FPX| 	^^ 	^
 
^D 15 ,0	"4!- %)	
  ,6 <@ +/	 '!8 $(	
 &,IV!F)4V||4{z	B 
6.(. 1. 
	.f #"	@@ @ 	@
 
@>0d0Qf3"1"5n<@7r@'R`DGR?BDL2)hO"G r;   r
  __main__)r8   rR   r   rR   )rB   rR   r   rR   )rG   zstream.Streamr   r  r   )rS   zt.AnyrT   r   rU   rR   rV   z
str | NonerK   z!t.Callable[[t.Any], t.Any] | NonerL   r  r   r  )Sr   
__future__r   collectionsr   r   r  r  r
  r  typingr  r   xml.etree.ElementTreer   r   r   r?   music21r   r	   r
   r   r   r   music21.common.enumsr   music21.common.numberToolsr   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r    r!   r"   music21.stream.iteratorr#   r$   r%   r&   r'   music21.musicxmlr(   "music21.musicxml.partStaffExporterr)   r*   music21.musicxml.xmlObjectsr,   r-   Environmentr  synchronizeIdsToXMLr  setXMLAttributeFromAttributer  r  music21.common.typesr.   r/   r0   r:   rC   rJ   rX   rZ   r  r   r!  r
  r   mainTestr   r;   r9   <module>r7     s   # #    	    
  "      / -                    2     $ E ' ? 7&&':;,,#@@  ??-!&R&J #	A 26AAA 
A 	A /A A AHV% V%r@
@ @
@JfO%; fV%W ? W xqGo qGlO z r;   