music21.layout

The layout.py module contains two types of objects that specify the layout on page (or screen) for Scores and other Stream objects. There are two main types of Layout objects: (1) layout describing elements and (2) layout defining Streams.

(1) ScoreLayout, PageLayout, SystemLayout, and StaffLayout objects describe the size of pages, the geometry of page and system margins, the distance between staves, etc. The model for these layout objects is taken directly (perhaps too directly?) from MusicXML. These objects all inherit from a BaseLayout class, primarily as an aid to finding all of these objects as a group. ScoreLayouts give defaults for each page, system, and staff. Thus, they contain PageLayout, SystemLayout, and currently one or more StaffLayout objects (but probably just one. MusicXML allows more than StaffLayout object because multiple staves can be in a Part. Music21 uses the concept of a PartStaff for a Part that is played by the same performer as another. e.g., the left hand of the Piano is a PartStaff paired with the right hand).

PageLayout and SystemLayout objects also have a property, ‘isNew’, which, if set to True, signifies that a new page or system should begin here. In theory, one could define new dimensions for a page or system in the middle of the system or page without setting isNew to True, in which case these measurements would start applying on the next page. In practice, there’s really one good place to use these Layout objects and that’s in the first part in a score at offset 0 of the first measure on a page or system (or for ScoreLayout, at the beginning of a piece outside any parts). But it’s not an error to put them in other places, such as at offset 0 of the first measure of a page or system in all the other parts. In fact, MusicXML tends to do this, and it ends up not being a waste if a program extracts a single part from the middle of a score.

These objects are standard Music21Object types, but many properties such as .duration, .beat, will probably not apply.

When exporting to MusicXML (which is currently the only format in which music21 can and does preserve these markings), many MusicXML readers will ignore these tags (or worse, add a new page or system when PageLayout and SystemLayout objects are found but also add theme wherever they want). In Finale, this behavior disappears if the MusicXML document notes that it <supports> new-page and new-system markings. Music21 will add the appropriate <supports> tags if the containing Stream has .definesExplicitPageBreaks and .definesExplicitSystemBreaks set to True. When importing a score that has the <supports> tag set, music21 will set .definesExplicitXXXXBreaks to True for the outer score and the inner parts. However, this means that if the score is manipulated enough that the prior layout information is obsolete, programs will need to set these properties to False or move the Layout tags accordingly.

(2) The second set of objects are Stream subclasses that can be employed when a program needs to easily iterate around the systems and pages defined through the layout objects just described, or to get the exact position on a page (or a graphical representation of a page) for a particular measure or system. (Individual notes coming soon). Normal Score streams can be changed into LayoutStreams by calling divideByPages(s) on them. A Score that was organized: Score->Part->Measure would then become: LayoutScore->Page->System->Staff->Measure.

The new LayoutScore has methods that enable querying what page or system a measure is in, and specifically where on a page a measure is (or the dimensions of every measure in the piece). However do not call .show() on a LayoutScore – the normal score it’s derived from will work just fine. Nor does calling .show() on a Page or System work yet, but once the LayoutStream has been created, code like this can be done:

s = stream.Stream(…) ls = layout.divideByPages(s) pg2sys3 = ls.pages[1].systems[2] # n.b.! 1, 2 measureStart, measureEnd = pg2sys3.measureStart, pg2sys3.measureEnd scoreExcerpt = s.measures(measureStart, measureEnd) scoreExcerpt.show() # will show page 2, system 3

Note that while the coordinates given by music21 for a musicxml score (based on margins, staff size, etc.) generally reflect what is actually in a musicxml producer, unfortunately, x-positions are far less accurately produced by most editors. For instance, Finale scores with measure sizes that have been manually adjusted tend to show their unadjusted measure width and not their actual measure width in the MusicXML.

SmartScore Pro tends to produce very good MusicXML layout data.

ScoreLayout

class music21.layout.ScoreLayout(*, scalingMillimeters: int | float | None = None, scalingTenths: int | float | None = None, musicFont: str | None = None, wordFont: str | None = None, pageLayout: PageLayout | None = None, systemLayout: SystemLayout | None = None, staffLayoutList: list[music21.layout.StaffLayout] | None = None, **keywords)

Parameters for configuring a score’s layout.

PageLayout objects may be found on Measure or Part Streams.

>>> pl = layout.PageLayout(pageNumber=4, leftMargin=234, rightMargin=124,
...                        pageHeight=4000, pageWidth=3000, isNew=True)
>>> pl.pageNumber
4
>>> pl.rightMargin
124
>>> pl.leftMargin
234
>>> pl.isNew
True

This object represents both <print new-page> and <page-layout> elements in musicxml. The appearance tag is handled in the .style for the stream (it was here in v7 and before, but did nothing).

Note that the appearance and style elements are subject to change during and after the v8 releases.

ScoreLayout bases

ScoreLayout read-only properties

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

ScoreLayout read/write properties

Read/write properties inherited from Music21Object:

ScoreLayout methods

ScoreLayout.tenthsToMillimeters(tenths)

given the scalingMillimeters and scalingTenths, return the value in millimeters of a number of musicxml “tenths” where a tenth is a tenth of the distance from one staff line to another

returns 0.0 if either of scalingMillimeters or scalingTenths is undefined.

>>> sl = layout.ScoreLayout(scalingMillimeters=2.0, scalingTenths=10)
>>> print(sl.tenthsToMillimeters(10))
2.0
>>> print(sl.tenthsToMillimeters(17))  # printing to round
3.4

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

ScoreLayout instance variables

Instance variables inherited from Music21Object:

PageLayout

class music21.layout.PageLayout(*, pageNumber: int | None = None, leftMargin: int | float | None = None, rightMargin: int | float | None = None, topMargin: int | float | None = None, bottomMargin: int | float | None = None, pageHeight: int | float | None = None, pageWidth: int | float | None = None, isNew: bool | None = None, **keywords)

Parameters for configuring a page’s layout.

PageLayout objects may be found on Measure or Part Streams.

>>> pl = layout.PageLayout(pageNumber=4, leftMargin=234, rightMargin=124,
...                        pageHeight=4000, pageWidth=3000, isNew=True)
>>> pl.pageNumber
4
>>> pl.rightMargin
124
>>> pl.leftMargin
234
>>> pl.isNew
True

This object represents both <print new-page> and <page-layout> elements in musicxml.

PageLayout bases

PageLayout read-only properties

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

PageLayout read/write properties

Read/write properties inherited from Music21Object:

PageLayout methods

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

PageLayout instance variables

Instance variables inherited from Music21Object:

SystemLayout

class music21.layout.SystemLayout(*, leftMargin: int | float | None = None, rightMargin: int | float | None = None, distance: int | float | None = None, topDistance: int | float | None = None, isNew: bool | None = None, **keywords)

Object that configures or alters a system’s layout.

SystemLayout objects may be found on Measure or Part Streams.

Importantly, if isNew is True then this object indicates that a new system should start here.

>>> sl = layout.SystemLayout(leftMargin=234, rightMargin=124, distance=3, isNew=True)
>>> sl.distance
3
>>> sl.rightMargin
124
>>> sl.leftMargin
234
>>> sl.isNew
True

SystemLayout bases

SystemLayout read-only properties

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

SystemLayout read/write properties

Read/write properties inherited from Music21Object:

SystemLayout methods

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

SystemLayout instance variables

Instance variables inherited from Music21Object:

StaffLayout

class music21.layout.StaffLayout(*, distance: int | float | None = None, staffNumber: int | float | None = None, staffSize: int | float | None = None, staffLines: int | None = None, hidden: bool | None = None, staffType: StaffType = StaffType.REGULAR, **keywords)

Object that configures or alters the distance between one staff and another in a system.

StaffLayout objects may be found on Measure or Part Streams.

The musicxml equivalent <staff-layout> lives in the <defaults> and in <print> attributes.

>>> sl = layout.StaffLayout(distance=3, staffNumber=1, staffSize=113, staffLines=5)
>>> sl.distance
3

The “number” attribute refers to which staff number in a part group this refers to. Thus, it’s not necessary in music21, but we store it if it’s there. (defaults to None)

>>> sl.staffNumber
1

staffLines specifies the number of lines for a non 5-line staff.

>>> sl.staffLines
5

staffSize is a percentage of the base staff size, so this defines a staff 13% larger than normal. Note that it is always converted to a floating point number.

>>> sl.staffSize
113.0
>>> sl
<music21.layout.StaffLayout distance 3, staffNumber 1, staffSize 113.0, staffLines 5>

StaffLayout can also specify the staffType:

>>> sl.staffType = stream.enums.StaffType.OSSIA

There is one other attribute, ‘.hidden’ which has three settings:

  • None - inherit from previous StaffLayout object, or False if no object exists

  • False - not hidden – show as a default staff

  • True - hidden – for playback only staves, or for a hidden/optimized-out staff

Note: (TODO: .hidden None is not working; always gives False)

StaffLayout bases

StaffLayout read-only properties

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

StaffLayout read/write properties

Read/write properties inherited from Music21Object:

StaffLayout methods

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

StaffLayout instance variables

StaffLayout.staffType

What kind of staff is this as a stream.enums.StaffType.

>>> sl = layout.StaffLayout()
>>> sl.staffType
<StaffType.REGULAR: 'regular'>
>>> sl.staffType = stream.enums.StaffType.CUE
>>> sl.staffType
<StaffType.CUE: 'cue'>

Instance variables inherited from Music21Object:

LayoutBase

class music21.layout.LayoutBase(id: str | int | None = None, groups: Groups | None = None, sites: Sites | None = None, duration: Duration | None = None, activeSite: stream.Stream | None = None, style: Style | None = None, editorial: Editorial | None = None, offset: OffsetQL = 0.0, quarterLength: OffsetQLIn | None = None, **keywords)

A base class for all Layout objects, defining a classSortOrder and also an inheritance tree.

>>> scoreLayout = layout.ScoreLayout()
>>> isinstance(scoreLayout, layout.LayoutBase)
True

LayoutBase bases

LayoutBase read-only properties

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

LayoutBase read/write properties

Read/write properties inherited from Music21Object:

LayoutBase methods

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

LayoutBase instance variables

Instance variables inherited from Music21Object:

LayoutScore

class music21.layout.LayoutScore(givenElements=None, **keywords)

Designation that this Score is divided into Pages, Systems, Staves (=Parts), Measures, etc.

Used for computing location of notes, etc.

If the score does not change between calls to the various getPosition calls, it is much faster as it uses a cache.

LayoutScore bases

LayoutScore read-only properties

LayoutScore.pages

Read-only properties inherited from Opus:

Read-only properties inherited from Stream:

Read-only properties inherited from StreamCore:

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

LayoutScore read/write properties

Read/write properties inherited from Stream:

Read/write properties inherited from Music21Object:

LayoutScore methods

LayoutScore.getAllMeasurePositionsInDocument(returnFormat='tenths', printUpdates=False)

returns a list of dictionaries, where each dictionary gives the measure number and other information, etc. in the document.

# >>> g = corpus.parse(‘luca/gloria’) # >>> gl = layout.divideByPages(g) # >>> gl.getAllMeasurePositionsInDocument()

LayoutScore.getMarginsAndSizeForPageId(pageId)

return a namedtuple of (top, left, bottom, right, width, height) margins for a given pageId in tenths

Default of (100, 100, 100, 100, 850, 1100) if undefined

>>> g = corpus.parse('luca/gloria')
>>> m22 = g.parts[0].getElementsByClass(stream.Measure)[22]
>>> m22.getElementsByClass(layout.PageLayout).first().leftMargin = 204.0
>>> gl = layout.divideByPages(g)
>>> gl.getMarginsAndSizeForPageId(1)
PageSize(top=171.0, left=204.0, right=171.0, bottom=171.0, width=1457.0, height=1886.0)
LayoutScore.getPageAndSystemNumberFromMeasureNumber(measureNumber)

Given a layoutScore from divideByPages and a measureNumber returns a tuple of (pageId, systemId). Note that pageId is probably one less than the page number, assuming that the first page number is 1, the pageId for the first page will be 0.

Similarly, the first systemId on each page will be 0

>>> lt = corpus.parse('demos/layoutTest.xml')
>>> l = layout.divideByPages(lt, fastMeasures=True)
>>> l.getPageAndSystemNumberFromMeasureNumber(80)
(3, 3)
LayoutScore.getPositionForStaff(pageId, systemId, staffId)

return a tuple of (top, bottom) for a staff, specified by a given pageId, systemId, and staffId in tenths of a staff-space.

This distance is specified with respect to the top of the system.

Staff scaling (<staff-details> in musicxml inside an <attributes> object) is taken into account, but not non-five-line staves. Thus, a normally sized staff is always of height 40 (4 spaces of 10-tenths each)

>>> lt = corpus.parse('demos/layoutTest.xml')
>>> ls = layout.divideByPages(lt, fastMeasures=True)

The first staff (staff 0) of each page/system always begins at height 0 and should end at height 40 if it is a 5-line staff (not taken into account) with no staffSize changes

>>> ls.getPositionForStaff(0, 0, 0)
(0.0, 40.0)
>>> ls.getPositionForStaff(1, 0, 0)
(0.0, 40.0)

The second staff (staff 1) begins at the end of staff 0 (40.0) + the appropriate staffDistance and adds the height of the staff. Staff 1 here has a size of 80 which means 80% of the normal staff size. 40 * 0.8 = 32.0:

>>> ls.getPositionForStaff(0, 0, 1)
(133.0, 165.0)

The third staff (staff 2) begins after the second staff (staff 1) but is a normal size staff

>>> ls.getPositionForStaff(0, 0, 2)
(266.0, 306.0)

The first staff (staff 0) of the second system (system 1) also begins at 0 and as a normally-sized staff, has height of 40:

>>> ls.getPositionForStaff(0, 1, 0)
(0.0, 40.0)

The spacing between the staves has changed in the second system, but the staff height has not:

>>> ls.getPositionForStaff(0, 1, 1)
(183.0, 215.0)
>>> ls.getPositionForStaff(0, 1, 2)
(356.0, 396.0)

In the third system (system 2), the staff distance reverts to the distance of system 0, but the staffSize is now 120 or 48 tenths (40 * 1.2 = 48)

>>> ls.getPositionForStaff(0, 2, 1)
(117.0, 165.0)

Page 1 (0), System 4 (3), Staff 2 (1) is a hidden (“optimized”) system. Thus, its staffLayout notes this:

>>> staffLayout031 = ls.pages[0].systems[3].staves[1].staffLayout
>>> staffLayout031
<music21.layout.StaffLayout distance None, staffNumber None, staffSize 80, staffLines None>
>>> staffLayout031.hidden
True

Thus, the position for this staff will have the same top and bottom, and the position for the next staff will have the same top as the previous staff:

>>> ls.getPositionForStaff(0, 3, 0)
(0.0, 40.0)
>>> ls.getPositionForStaff(0, 3, 1)
(40.0, 40.0)
>>> ls.getPositionForStaff(0, 3, 2)
(133.0, 173.0)

Tests for a score with PartStaff objects: >>> lt = corpus.parse(‘demos/layoutTestMore.xml’) >>> ls = layout.divideByPages(lt, fastMeasures = True) >>> ls.getPositionForStaff(0, 0, 0) (0.0, 40.0) >>> ls.getPositionForStaff(0, 0, 1) (133.0, 173.0) >>> ls.getPositionForStaff(0, 0, 2) (235.0, 275.0)

>>> ls.getPositionForStaff(0, 2, 0)
(0.0, 40.0)
>>> ls.getPositionForStaff(0, 2, 1)
(40.0, 40.0)
>>> ls.getPositionForStaff(0, 2, 2)
(40.0, 40.0)

System 4 has the top staff hidden, which has been causing problems:

>>> ls.getPositionForStaff(0, 4, 0)
(0.0, 0.0)
>>> ls.getPositionForStaff(0, 4, 1)
(0.0, 40.0)
LayoutScore.getPositionForStaffMeasure(staffId, measureNumber, returnFormat='tenths')

Given a layoutScore from divideByPages, a staffId, and a measureNumber, returns a tuple of ((top, left), (bottom, right), pageId) allowing an exact position for the measure on the page. If returnFormat is “tenths”, then it will be returned in tenths.

If returnFormat is “float”, returns each as a number from 0 to 1 where 0 is the top or left of the page, and 1 is the bottom or right of the page.

>>> lt = corpus.parse('demos/layoutTest.xml')
>>> ls = layout.divideByPages(lt, fastMeasures = True)

The first measure of staff one begins at 336 tenths from the top (125 for the margin top and 211 for the top-staff-distance). It begins 170.0 from the left (100 for the page-margin-left, 70 for staff-margin-left). It ends 40.0 below that (staffHeight) and 247.0 to the right (measure width)

>>> ls.getPositionForStaffMeasure(0, 1)
((336.0, 170.0), (376.0, 417.0), 0)

The other staves for the same measure are below this one:

>>> ls.getPositionForStaffMeasure(1, 1)
((469.0, 170.0), (501.0, 417.0), 0)
>>> ls.getPositionForStaffMeasure(2, 1)
((602.0, 170.0), (642.0, 417.0), 0)

If float is requested for returning, then the numbers are the fraction of the distance across the page.

>>> ls.getPositionForStaffMeasure(0, 1, returnFormat='float')
((0.152..., 0.0996...), (0.170..., 0.244...), 0)

Moving over the page boundary:

>>> ls.getPositionForStaffMeasure(0, 23)
((1703.0, 1345.0), (1743.0, 1606.0), 0)
>>> ls.getPositionForStaffMeasure(1, 23)  # hidden
((1743.0, 1345.0), (1743.0, 1606.0), 0)
>>> ls.getPositionForStaffMeasure(0, 24)
((195.0, 100.0), (235.0, 431.0), 1)
>>> ls.getPositionForStaffMeasure(1, 24)
((328.0, 100.0), (360.0, 431.0), 1)
LayoutScore.getPositionForSystem(pageId, systemId)

first systems on a page use a different positioning.

returns a Named tuple of the (top, left, right, and bottom) where each unit is relative to the page margins

N.B. right is NOT the width – it is different. It is the offset to the right margin. weird, inconsistent, but most useful…bottom is the hard part to compute…

>>> lt = corpus.parse('demos/layoutTestMore.xml')
>>> ls = layout.divideByPages(lt, fastMeasures = True)
>>> ls.getPositionForSystem(0, 0)
SystemSize(top=211.0, left=70.0, right=0.0, bottom=696.0)
>>> ls.getPositionForSystem(0, 1)
SystemSize(top=810.0, left=0.0, right=0.0, bottom=1173.0)
>>> ls.getPositionForSystem(0, 2)
SystemSize(top=1340.0, left=67.0, right=92.0, bottom=1610.0)
>>> ls.getPositionForSystem(0, 3)
SystemSize(top=1724.0, left=0.0, right=0.0, bottom=2030.0)
>>> ls.getPositionForSystem(0, 4)
SystemSize(top=2144.0, left=0.0, right=0.0, bottom=2583.0)
LayoutScore.getStaffDistanceFromPrevious(pageId, systemId, staffId)

return the distance of this staff from the previous staff in the same system

for staffId = 0, this is always 0.0

TODO:tests, now that this is out from previous

LayoutScore.getStaffHiddenAttribute(pageId: int, systemId: int, staffId: int) bool

returns the staffLayout.hidden attribute for a staffId, or if it is not defined, recursively search through previous staves until one is found.

>>> lt = corpus.parse('demos/layoutTestMore.xml')
>>> ls = layout.divideByPages(lt, fastMeasures = True)
>>> ls.getStaffHiddenAttribute(0, 0, 0)
False
>>> ls.getStaffHiddenAttribute(0, 0, 1)
False
>>> ls.getStaffHiddenAttribute(0, 1, 1)
True
>>> ls.getStaffHiddenAttribute(0, 2, 1)
True
>>> ls.getStaffHiddenAttribute(0, 3, 1)
False
LayoutScore.getStaffSizeFromLayout(pageId: int, systemId: int, staffId: int) float

Get the currently active staff-size for a given pageId, systemId, and staffId.

Note that this does not take into account the hidden state of the staff, which, if True, makes the effective size 0.0 – see getStaffHiddenAttribute

>>> lt = corpus.parse('demos/layoutTest.xml')
>>> ls = layout.divideByPages(lt, fastMeasures=True)
>>> ls.getStaffSizeFromLayout(0, 0, 0)
40.0
>>> ls.getStaffSizeFromLayout(0, 0, 1)
32.0
>>> ls.getStaffSizeFromLayout(0, 0, 2)
40.0
>>> ls.getStaffSizeFromLayout(0, 1, 1)
32.0
>>> ls.getStaffSizeFromLayout(0, 2, 1)
48.0
>>> ls.getStaffSizeFromLayout(0, 3, 1)
32.0
LayoutScore.getSystemBeforeThis(pageId: int, systemId: int) tuple[int | None, int]

given a pageId and systemId, get the (pageId, systemId) for the previous system.

return (None, None) if it’s the first system on the first page

This test score has five systems on the first page, three on the second, and two on the third

>>> lt = corpus.parse('demos/layoutTestMore.xml')
>>> ls = layout.divideByPages(lt, fastMeasures = True)
>>> systemId = 1
>>> pageId = 2  # last system, last page
>>> while pageId is not None:
...    pageId, systemId = ls.getSystemBeforeThis(pageId, systemId)
...    (pageId, systemId)
(2, 0) (1, 2) (1, 1) (1, 0) (0, 4) (0, 3) (0, 2) (0, 1) (0, 0) (None, -1)
LayoutScore.measurePositionWithinSystem(measureNumber, pageId=None, systemId=None)

Given a measure number, find the start and end X positions (with respect to the system margins) for the measure.

if pageId and systemId are given, then it will speed up the search. But not necessary

no staffId is needed since (at least for now) all measures begin and end at the same X position

>>> l = corpus.parse('demos/layoutTest.xml')
>>> ls = layout.divideByPages(l, fastMeasures = True)
>>> ls.measurePositionWithinSystem(1, 0, 0)
(0.0, 247.0)
>>> ls.measurePositionWithinSystem(2, 0, 0)
(247.0, 544.0)
>>> ls.measurePositionWithinSystem(3, 0, 0)
(544.0, 841.0)

Measure positions reset at the start of a new system

>>> ls.measurePositionWithinSystem(6)
(0.0, 331.0)
>>> ls.measurePositionWithinSystem(7)
(331.0, 549.0)
LayoutScore.show(fmt=None, app=None, **keywords)

Borrows stream.Score.show

>>> lp = layout.Page()
>>> ls = layout.LayoutScore()
>>> ls.append(lp)
>>> ls.show('text')
{0.0} <music21.layout.Page p.1>

Methods inherited from Opus:

Methods inherited from Stream:

Methods inherited from StreamCore:

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

LayoutScore instance variables

Instance variables inherited from Stream:

Instance variables inherited from Music21Object:

Page

class music21.layout.Page(givenElements=None, **keywords)

Designation that all the music in this Stream belongs on a single notated page.

Page bases

Page read-only properties

Page.systems

Read-only properties inherited from Opus:

Read-only properties inherited from Stream:

Read-only properties inherited from StreamCore:

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

Page read/write properties

Read/write properties inherited from Stream:

Read/write properties inherited from Music21Object:

Page methods

Page.show(fmt=None, app=None, **keywords)

Borrows stream.Score.show

>>> ls = layout.System()
>>> lp = layout.Page()
>>> lp.append(ls)
>>> lp.show('text')
{0.0} <music21.layout.System 0: p.0, sys.0>

Methods inherited from Opus:

Methods inherited from Stream:

Methods inherited from StreamCore:

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

Page instance variables

Instance variables inherited from Stream:

Instance variables inherited from Music21Object:

System

class music21.layout.System(givenElements=None, **keywords)

Designation that all the music in this Stream belongs on a single notated system.

Attribute systemNumbering says at what point the numbering of systems resets. It can be either “Score” (default), “Opus”, or “Page”.

System bases

System read-only properties

System.staves

Read-only properties inherited from Score:

Read-only properties inherited from Stream:

Read-only properties inherited from StreamCore:

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

System read/write properties

Read/write properties inherited from Stream:

Read/write properties inherited from Music21Object:

System methods

Methods inherited from Score:

Methods inherited from Stream:

Methods inherited from StreamCore:

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

System instance variables

Instance variables inherited from Stream:

Instance variables inherited from Music21Object:

Staff

class music21.layout.Staff(givenElements=None, **keywords)

Designation that all the music in this Stream belongs on a single Staff.

Staff bases

Staff read-only properties

Read-only properties inherited from Stream:

Read-only properties inherited from StreamCore:

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

Staff read/write properties

Read/write properties inherited from Part:

Read/write properties inherited from Stream:

Read/write properties inherited from Music21Object:

Staff methods

Methods inherited from Part:

Methods inherited from Stream:

Methods inherited from StreamCore:

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

Staff instance variables

Instance variables inherited from Stream:

Instance variables inherited from Music21Object:

PageSize

class music21.layout.PageSize(top, left, right, bottom, width, height)

StaffGroup

class music21.layout.StaffGroup(*spannedElements, name: str | None = None, barTogether: Literal[True, False, None, 'Mensurstrich'] = True, abbreviation: str | None = None, symbol: Literal['bracket', 'line', 'brace', 'square'] | None = None, **keywords)

A StaffGroup defines a collection of one or more Part objects, specifying that they should be shown together with a bracket, brace, or other symbol, and may have a common name.

>>> p1 = stream.Part()
>>> p2 = stream.Part()
>>> p1.append(note.Note('C5', type='whole'))
>>> p1.append(note.Note('D5', type='whole'))
>>> p2.append(note.Note('C3', type='whole'))
>>> p2.append(note.Note('D3', type='whole'))
>>> p3 = stream.Part()
>>> p3.append(note.Note('F#4', type='whole'))
>>> p3.append(note.Note('G#4', type='whole'))
>>> s = stream.Score()
>>> s.insert(0, p1)
>>> s.insert(0, p2)
>>> s.insert(0, p3)
>>> staffGroup1 = layout.StaffGroup([p1, p2],
...      name='Marimba', abbreviation='Mba.', symbol='brace')
>>> staffGroup1.barTogether = 'Mensurstrich'
>>> s.insert(0, staffGroup1)
>>> staffGroup2 = layout.StaffGroup([p3],
...      name='Xylophone', abbreviation='Xyl.', symbol='bracket')
>>> s.insert(0, staffGroup2)
>>> s.show()
../_images/layout_StaffGroup_01.jpg

StaffGroup bases

StaffGroup read-only properties

Read-only properties inherited from Music21Object:

Read-only properties inherited from ProtoM21Object:

StaffGroup read/write properties

StaffGroup.barTogether

Get or set the barTogether value, with either Boolean values or yes or no strings. Or the string ‘Mensurstrich’ which indicates barring between staves but not in staves.

Currently Mensurstrich is not supported by most exporters.

>>> sg = layout.StaffGroup()
>>> sg.barTogether = 'yes'
>>> sg.barTogether
True
>>> sg.barTogether = 'Mensurstrich'
>>> sg.barTogether
'Mensurstrich'
StaffGroup.symbol

Get or set the symbol value, with either Boolean values or yes or no strings.

>>> sg = layout.StaffGroup()
>>> sg.symbol = 'Brace'
>>> sg.symbol
'brace'

Read/write properties inherited from Music21Object:

StaffGroup methods

Methods inherited from Spanner:

Methods inherited from Music21Object:

Methods inherited from ProtoM21Object:

StaffGroup instance variables

Instance variables inherited from Music21Object:

SystemSize

class music21.layout.SystemSize(top, left, right, bottom)

Functions

music21.layout.divideByPages(scoreIn: Score, printUpdates: bool = False, fastMeasures: bool = False) LayoutScore

Divides a score into a series of smaller scores according to page breaks. Only searches for PageLayout.isNew or SystemLayout.isNew on the first part. Returns a new LayoutScore object.

If fastMeasures is True, then the newly created System objects do not have Clef signs, Key Signatures, or necessarily all the applicable spanners in them. On the other hand, the position (on the page) information will be just as correct with fastMeasures = True and it will run much faster on large scores (because our spanner gathering algorithm is currently O(n^2); something TODO: to fix.)

>>> lt = corpus.parse('demos/layoutTest.xml')
>>> len(lt.parts)
3
>>> len(lt.parts[0].getElementsByClass(stream.Measure))
80

Divide the score up into layout.Page objects

>>> layoutScore = layout.divideByPages(lt, fastMeasures=True)
>>> len(layoutScore.pages)
4
>>> lastPage = layoutScore.pages[-1]
>>> lastPage.measureStart
64
>>> lastPage.measureEnd
80

the layoutScore is a subclass of stream.Opus:

>>> layoutScore
<music21.layout.LayoutScore ...>
>>> 'Opus' in layoutScore.classes
True

Pages are subclasses of Opus also, since they contain Scores

>>> lastPage
<music21.layout.Page ...>
>>> 'Opus' in lastPage.classes
True

Each page now has Systems not parts.

>>> firstPage = layoutScore.pages[0]
>>> len(firstPage.systems)
4
>>> firstSystem = firstPage.systems[0]
>>> firstSystem.measureStart
1
>>> firstSystem.measureEnd
5

Systems are a subclass of Score:

>>> firstSystem
<music21.layout.System ...>
>>> isinstance(firstSystem, stream.Score)
True

Each System has staves (layout.Staff objects) not parts, though Staff is a subclass of Part

>>> secondStaff = firstSystem.staves[1]
>>> print(len(secondStaff.getElementsByClass(stream.Measure)))
5
>>> secondStaff
<music21.layout.Staff ...>
>>> isinstance(secondStaff, stream.Part)
True
music21.layout.getPageRegionMeasureNumbers(scoreIn)
music21.layout.getRegionMeasureNumbers(scoreIn, region='Page')

get a list where each entry is a 2-tuplet whose first number refers to the first measure on a page and whose second number is the last measure on the page.

music21.layout.getSystemRegionMeasureNumbers(scoreIn)