"""Music class.
This module defines the core class of MusPy---the Music class, a
universal container for symbolic music.
Classes
-------
- Music
Variables
---------
- DEFAULT_RESOLUTION
"""
from collections import OrderedDict
from math import ceil, floor
from pathlib import Path
from typing import Any, Callable, List, Union
import numpy as np
from mido import MidiFile
from music21.stream import Stream
from numpy import ndarray
from pretty_midi import PrettyMIDI
from pypianoroll import Multitrack
from .base import ComplexBase
from .classes import (
Annotation,
Beat,
KeySignature,
Lyric,
Metadata,
Tempo,
TimeSignature,
Track,
)
from .outputs import save, synthesize, to_object, to_representation, write
from .visualization import show
__all__ = ["Music", "DEFAULT_RESOLUTION"]
DEFAULT_RESOLUTION = 24
# pylint: disable=super-init-not-called
[docs]class Music(ComplexBase):
"""A universal container for symbolic music.
This is the core class of MusPy. A Music object can be constructed
in the following ways.
- :meth:`muspy.Music`: Construct by setting values for attributes.
- :meth:`muspy.Music.from_dict`: Construct from a dictionary that
stores the attributes and their values as key-value pairs.
- :func:`muspy.read`: Read from a MIDI, a MusicXML or an ABC file.
- :func:`muspy.load`: Load from a JSON or a YAML file saved by
:func:`muspy.save`.
- :func:`muspy.from_object`: Convert from a `music21.Stream`, a
:class:`mido.MidiFile`, a :class:`pretty_midi.PrettyMIDI` or a
:class:`pypianoroll.Multitrack` object.
Attributes
----------
metadata : :class:`muspy.Metadata`, default: `Metadata()`
Metadata.
resolution : int, default: `muspy.DEFAULT_RESOLUTION` (24)
Time steps per quarter note.
tempos : list of :class:`muspy.Tempo`, default: []
Tempo changes.
key_signatures : list of :class:`muspy.KeySignature`, default: []
Key signatures changes.
time_signatures : list of :class:`muspy.TimeSignature`, default: []
Time signature changes.
beats : list of :class:`muspy.Beat`, default: []
Beats.
lyrics : list of :class:`muspy.Lyric`, default: []
Lyrics.
annotations : list of :class:`muspy.Annotation`, default: []
Annotations.
tracks : list of :class:`muspy.Track`, default: []
Music tracks.
Note
----
Indexing a Music object returns the track of a certain index. That
is, ``music[idx]`` returns ``music.tracks[idx]``. Length of a Music
object is the number of tracks. That is, ``len(music)`` returns
``len(music.tracks)``.
"""
_attributes = OrderedDict(
[
("metadata", Metadata),
("resolution", int),
("tempos", Tempo),
("key_signatures", KeySignature),
("time_signatures", TimeSignature),
("beats", Beat),
("lyrics", Lyric),
("annotations", Annotation),
("tracks", Track),
]
)
_optional_attributes = [
"metadata",
"resolution",
"tempos",
"key_signatures",
"time_signatures",
"beats",
"lyrics",
"annotations",
"tracks",
]
_list_attributes = [
"tempos",
"key_signatures",
"time_signatures",
"beats",
"lyrics",
"annotations",
"tracks",
]
def __init__(
self,
metadata: Metadata = None,
resolution: int = None,
tempos: List[Tempo] = None,
key_signatures: List[KeySignature] = None,
time_signatures: List[TimeSignature] = None,
beats: List[Beat] = None,
lyrics: List[Lyric] = None,
annotations: List[Annotation] = None,
tracks: List[Track] = None,
):
self.metadata = metadata if metadata is not None else Metadata()
self.resolution = (
resolution if resolution is not None else DEFAULT_RESOLUTION
)
self.tempos = tempos if tempos is not None else []
self.key_signatures = (
key_signatures if key_signatures is not None else []
)
self.time_signatures = (
time_signatures if time_signatures is not None else []
)
self.beats = beats if beats is not None else []
self.lyrics = lyrics if lyrics is not None else []
self.annotations = annotations if annotations is not None else []
self.tracks = tracks if tracks is not None else []
def __len__(self) -> int:
return len(self.tracks)
def __getitem__(self, key: int) -> Track:
return self.tracks[key]
def __setitem__(self, key: int, value: Track):
self.tracks[key] = value
[docs] def get_end_time(self, is_sorted: bool = False) -> int:
"""Return the the time of the last event in all tracks.
This includes tempos, key signatures, time signatures, note
offsets, lyrics and annotations.
Parameters
----------
is_sorted : bool, default: False
Whether all the list attributes are sorted.
"""
def _get_end_time(list_):
if not list_:
return 0
if is_sorted:
return list_[-1].time
return max(item.time for item in list_)
if self.tracks:
track_end_time = max(
track.get_end_time(is_sorted) for track in self.tracks
)
else:
track_end_time = 0
end_time = max(
_get_end_time(self.tempos),
_get_end_time(self.key_signatures),
_get_end_time(self.time_signatures),
_get_end_time(self.beats),
_get_end_time(self.lyrics),
_get_end_time(self.annotations),
track_end_time,
)
return end_time
[docs] def get_real_end_time(self, is_sorted: bool = False) -> float:
"""Return the end time in realtime.
This includes tempos, key signatures, time signatures, note
offsets, lyrics and annotations. Assume 120 qpm (quarter notes
per minute) if no tempo information is available.
Parameters
----------
is_sorted : bool, default: False
Whether all the list attributes are sorted.
"""
# Get symbolic end time
end_time = self.get_end_time(is_sorted=is_sorted)
# If no tempo information is available, assume 120 qpm
if not self.tempos:
return 0.5 * end_time / self.resolution
# Compute the real end time
position = 0.0
qpm = 120.0
factor = 60.0 / self.resolution
real_end_time = 0.0
for tempo in self.tempos:
real_end_time += (tempo.time - position) * factor / qpm
position = tempo.time
qpm = tempo.qpm
real_end_time += (end_time - position) * factor / qpm
return real_end_time
[docs] def infer_beats(self) -> List[Beat]:
"""Infer beats from the time signature changes.
This assumes that there is a downbeat at each time signature
change (this is not always true, e.g., for a pickup measure).
Returns
-------
list of :class:`muspy.Beat`
List of beats inferred from the time signature changes.
Return an empty list if no time signature is found.
"""
beats: List[Beat] = []
for i, time_sign in enumerate(self.time_signatures):
if i == len(self.time_signatures) - 1:
end = self.get_end_time()
else:
end = self.time_signatures[i + 1].time
beat_resolution = self.resolution / (time_sign.denominator / 4)
for j, time in enumerate(
np.arange(time_sign.time, end, beat_resolution)
):
if j % time_sign.numerator == 0:
beats.append(Beat(time=round(time), is_downbeat=True))
else:
beats.append(Beat(time=round(time), is_downbeat=False))
return beats
[docs] def adjust_resolution(
self,
target: int = None,
factor: float = None,
rounding: Union[str, Callable] = "round",
) -> "Music":
"""Adjust resolution and timing of all time-stamped objects.
Parameters
----------
target : int, optional
Target resolution.
factor : int or float, optional
Factor used to adjust the resolution based on the formula:
`new_resolution = old_resolution * factor`. For example, a
factor of 2 double the resolution, and a factor of 0.5 halve
the resolution.
rounding : {'round', 'ceil', 'floor'} or callable, default:
'round'
Rounding mode.
Returns
-------
Object itself.
"""
if self.resolution is None:
raise TypeError("`resolution` must be given.")
if self.resolution < 0:
raise ValueError("`resolution` must be positive.")
if target is None and factor is None:
raise ValueError("One of `target` and `factor` must be given.")
if target is not None and factor is not None:
raise ValueError("Only one of `target` and `factor` can be given.")
if rounding is None or rounding == "round":
rounding = round
elif rounding == "ceil":
rounding = ceil
elif rounding == "floor":
rounding = floor
elif isinstance(rounding, str):
raise ValueError(f"Unrecognized rounding mode : {rounding} .")
if target is not None:
if not isinstance(target, int):
raise TypeError("`target` must be an integer.")
target_ = int(target)
factor_ = target / self.resolution
if factor is not None:
new_resolution = float(self.resolution * factor)
if not new_resolution.is_integer():
raise ValueError(
f"`factor` must be a factor of the original resolution "
f"{self.resolution}, but got : {factor}."
)
factor_ = float(factor)
target_ = int(new_resolution)
self.resolution = int(target_)
self.adjust_time(lambda time: rounding(time * factor_)) # type: ignore
return self
[docs] def clip(self, lower: int = 0, upper: int = 127) -> "Music":
"""Clip the velocity of each note for each track.
Parameters
----------
lower : int, default: 0
Lower bound.
upper : int, default: 127
Upper bound.
Returns
-------
Object itself.
"""
for track in self.tracks:
track.clip(lower, upper)
return self
[docs] def transpose(self, semitone: int) -> "Music":
"""Transpose all the notes by a number of semitones.
Parameters
----------
semitone : int
Number of semitones to transpose the notes. A positive value
raises the pitches, while a negative value lowers the
pitches.
Returns
-------
Object itself.
Notes
-----
Drum tracks are skipped.
"""
for track in self.tracks:
if not track.is_drum:
track.transpose(semitone)
return self
[docs] def save(self, path: Union[str, Path], kind: str = None, **kwargs: Any):
"""Save loselessly to a JSON or a YAML file.
Refer to :func:`muspy.save` for full documentation.
"""
return save(path, self, kind=kind, **kwargs)
[docs] def save_json(self, path: Union[str, Path], **kwargs: Any):
"""Save loselessly to a JSON file.
Refer to :func:`muspy.save_json` for full documentation.
"""
return save(path, self, kind="json", **kwargs)
[docs] def save_yaml(self, path: Union[str, Path]):
"""Save loselessly to a YAML file.
Refer to :func:`muspy.save_yaml` for full documentation.
"""
return save(path, self, kind="yaml")
[docs] def write(self, path: Union[str, Path], kind: str = None, **kwargs: Any):
"""Write to a MIDI, a MusicXML, an ABC or an audio file.
Refer to :func:`muspy.write` for full documentation.
"""
return write(path, self, kind=kind, **kwargs)
[docs] def write_midi(self, path: Union[str, Path], **kwargs: Any):
"""Write to a MIDI file.
Refer to :func:`muspy.write_midi` for full documentation.
"""
return write(path, self, kind="midi", **kwargs)
[docs] def write_musicxml(self, path: Union[str, Path], **kwargs: Any):
"""Write to a MusicXML file.
Refer to :func:`muspy.write_musicxml` for full documentation.
"""
return write(path, self, kind="musicxml", **kwargs)
[docs] def write_abc(self, path: Union[str, Path], **kwargs: Any):
"""Write to an ABC file.
Refer to :func:`muspy.write_abc` for full documentation.
"""
return write(path, self, kind="abc", **kwargs)
[docs] def write_audio(self, path: Union[str, Path], **kwargs: Any):
"""Write to an audio file.
Refer to :func:`muspy.write_audio` for full documentation.
"""
return write(path, self, kind="audio", **kwargs)
[docs] def to_object(self, kind: str, **kwargs: Any):
"""Return as an object in other libraries.
Refer to :func:`muspy.to_object` for full documentation.
"""
return to_object(self, kind=kind, **kwargs)
[docs] def to_music21(self, **kwargs: Any) -> Stream:
"""Return as a Stream object.
Refer to :func:`muspy.to_music21` for full documentation.
"""
return to_object(self, kind="music21", **kwargs)
[docs] def to_mido(self, **kwargs: Any) -> MidiFile:
"""Return as a MidiFile object.
Refer to :func:`muspy.to_mido` for full documentation.
"""
return to_object(self, kind="mido", **kwargs)
[docs] def to_pretty_midi(self, **kwargs: Any) -> PrettyMIDI:
"""Return as a PrettyMIDI object.
Refer to :func:`muspy.to_pretty_midi` for full documentation.
"""
return to_object(self, kind="pretty_midi", **kwargs)
[docs] def to_pypianoroll(self, **kwargs: Any) -> Multitrack:
"""Return as a Multitrack object.
Refer to :func:`muspy.to_pypianoroll` for full documentation.
"""
return to_object(self, kind="pypianoroll", **kwargs)
[docs] def to_representation(self, kind: str, **kwargs: Any) -> ndarray:
"""Return in a specific representation.
Refer to :func:`muspy.to_representation` for full documentation.
"""
return to_representation(self, kind=kind, **kwargs)
[docs] def to_pitch_representation(self, **kwargs: Any) -> ndarray:
"""Return in pitch-based representation.
Refer to :func:`muspy.to_pitch_representation` for full
documentation.
"""
return to_representation(self, kind="pitch", **kwargs)
[docs] def to_pianoroll_representation(self, **kwargs: Any) -> ndarray:
"""Return in piano-roll representation.
Refer to :func:`muspy.to_pianoroll_representation` for full
documentation.
"""
return to_representation(self, kind="piano-roll", **kwargs)
[docs] def to_event_representation(self, **kwargs: Any) -> ndarray:
"""Return in event-based representation.
Refer to :func:`muspy.to_event_representation` for full
documentation.
"""
return to_representation(self, kind="event", **kwargs)
[docs] def to_note_representation(self, **kwargs: Any) -> ndarray:
"""Return in note-based representation.
Refer to :func:`muspy.to_note_representation` for full
documentation.
"""
return to_representation(self, kind="note", **kwargs)
[docs] def show(self, kind: str, **kwargs: Any):
"""Show visualization.
Refer to :func:`muspy.show` for full documentation.
"""
return show(self, kind, **kwargs)
[docs] def show_score(self, **kwargs: Any):
"""Show score visualization.
Refer to :func:`muspy.show_score` for full documentation.
"""
return show(self, kind="score", **kwargs)
[docs] def show_pianoroll(self, **kwargs: Any):
"""Show pianoroll visualization.
Refer to :func:`muspy.show_pianoroll` for full documentation.
"""
return show(self, kind="piano-roll", **kwargs)
[docs] def synthesize(self, **kwargs) -> ndarray:
"""Synthesize a Music object to raw audio.
Refer to :func:`muspy.synthesize` for full documentation.
"""
return synthesize(self, **kwargs)