Metadata-Version: 2.4
Name: pianoplayer
Version: 3.0.2
Summary: Automatic piano fingering generator. Finds and shows in 3D the best fingering combination to play a score.
Author-email: Marco Musy <marco.musy@gmail.com>
License: MIT
Project-URL: Homepage, https://github.com/marcomusy/pianoplayer
Project-URL: Repository, https://github.com/marcomusy/pianoplayer
Keywords: piano,fingering,musicxml
Classifier: Intended Audience :: Science/Research
Classifier: Intended Audience :: Education
Classifier: Programming Language :: Python :: 3
Classifier: License :: OSI Approved :: MIT License
Classifier: Topic :: Scientific/Engineering :: Visualization
Classifier: Topic :: Scientific/Engineering :: Information Analysis
Requires-Python: >=3.9
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: rich>=13.0.0
Provides-Extra: visual
Requires-Dist: vedo>=2026.6.1; extra == "visual"
Provides-Extra: midi
Requires-Dist: pretty_midi>=0.2.9; extra == "midi"
Provides-Extra: sound
Requires-Dist: pygame>=2.6.0; extra == "sound"
Provides-Extra: build
Requires-Dist: pyinstaller>=6.0.0; extra == "build"
Provides-Extra: web
Requires-Dist: fastapi>=0.115.0; extra == "web"
Requires-Dist: uvicorn>=0.30.0; extra == "web"
Requires-Dist: python-multipart>=0.0.9; extra == "web"
Provides-Extra: all
Requires-Dist: vedo>=2026.6.1; extra == "all"
Requires-Dist: pretty_midi>=0.2.9; extra == "all"
Requires-Dist: pygame>=2.6.0; extra == "all"
Requires-Dist: pyinstaller>=6.0.0; extra == "all"
Requires-Dist: fastapi>=0.115.0; extra == "all"
Requires-Dist: uvicorn>=0.30.0; extra == "all"
Requires-Dist: python-multipart>=0.0.9; extra == "all"
Dynamic: license-file


![banner](https://github.com/user-attachments/assets/7242fc4e-873f-42fc-99ff-edf985a412eb)

[![PyPI version](https://img.shields.io/pypi/v/pianoplayer.svg)](https://pypi.org/project/pianoplayer/)
[![Downloads](https://pepy.tech/badge/pianoplayer)](https://pepy.tech/project/pianoplayer)
[![Downloads / Month](https://pepy.tech/badge/pianoplayer/month)](https://pepy.tech/project/pianoplayer)
[![lics](https://img.shields.io/badge/license-MIT-blue.svg)](https://en.wikipedia.org/wiki/MIT_License)
[![DOI](https://zenodo.org/badge/107160052.svg)](https://zenodo.org/badge/latestdoi/107160052)
[![CI](https://github.com/marcomusy/pianoplayer/actions/workflows/ci.yml/badge.svg)](https://github.com/marcomusy/pianoplayer/actions/workflows/ci.yml)
[![Stars](https://img.shields.io/github/stars/marcomusy/pianoplayer.svg?style=social)](https://github.com/marcomusy/pianoplayer/stargazers)


Automatic piano fingering generator for MusicXML and MIDI scores.  
PianoPlayer searches for a low-effort fingering sequence for one or both hands.  

## Download and Install:
```bash
pip install pianoplayer
```
Optional extras:
```bash
pip install "pianoplayer[visual]"  # 3D rendering with vedo
pip install "pianoplayer[midi]"    # MIDI input support
pip install "pianoplayer[sound]"   # enable playback
pip install "pianoplayer[all]"     # all optional extras
```

<details>
<summary><strong>Python Setup (Beginner-Friendly)</strong></summary>

If you are new to Python, use one of these two approaches.

### Option A: Anaconda (recommended for beginners on Windows)

1. Install Anaconda from https://www.anaconda.com/download
2. Type **Anaconda Prompt** in your windows search and open it.
3. Install PianoPlayer, type:
```bash
pip install pianoplayer
```
4. Run:
```bash
pianoplayer --help
```

### Option B: Python from python.org

Windows:
1. Install Python 3.10+ from https://www.python.org/downloads/windows/
2. During installation, enable **Add Python to PATH**.
3. Open **Command Prompt** and run:
```bash
python -m pip install --upgrade pip
pip install pianoplayer
pianoplayer --help
```

macOS/Linux:
```bash
python3 -m pip install --upgrade pip
pip install pianoplayer
pianoplayer --help
```

### Standalone executable (without 3D visualization)
Build a standalone executable with `PyInstaller`:

```bash
pip install "pianoplayer[build]"
python scripts/build_standalone.py
```

Output executable:
- Linux/macOS: `dist/pianoplayer`
- Windows: `dist/pianoplayer.exe`

</details>

To visualize the output annotated score (`output.xml`) install the latest
[musescore](https://musescore.org/en/download), or any other renderer
of [MusicXML](https://en.wikipedia.org/wiki/MusicXML)
files.
You can also inspect playback in 3D with [vedo](https://github.com/marcomusy/vedo).


## CLI Usage:
Example command line:
`pianoplayer scores/bach_invention4.xml -n 10 -r -v -z -m`  
This annotates the first 10 measures for the right hand, opens 3D playback, and then opens MuseScore.

The output is saved as a [MusicXML](https://en.wikipedia.org/wiki/MusicXML)
file with name `output.xml`.<br />

Pre-fingered notes are supported: if a note already has a fingering mark, `PianoPlayer` keeps it
and uses it as an anchor for the following optimization.
In the output score, these anchored fingers are rendered as circled numbers.

```bash
pianoplayer         # if no argument is given a GUI will pop up
# Or
pianoplayer [-h] [--gui] [-o] [-n] [-s] [-d] [-rpart] [-lpart] [--rstaff] [--lstaff]
            [--auto-routing | --manual-routing]
            [--quiet] [-m] [-b] [-v] [-z] [-l] [-r]
            [--hand-size {XXS,XS,S,M,L,XL,XXL}] [--chord-note-stagger-s]
            filename
# Valid file formats: MusicXML, compressed MusicXML, MuseScore, MIDI, PIG
# (.xml, .mxl, .mscz, .mscx, .mid, .midi, .txt)
#
# Optional arguments:
#   -h, --help            show this help message and exit
#   --gui                 Launch the Tkinter GUI
#   -o , --outputfile     Annotated output xml file name
#   -n , --n-measures     [1000] Number of score measures to scan
#   -s , --start-measure  Start from measure number [1]
#   -d , --depth          [auto] Depth of combinatorial search, [5-9]
#   -rpart                [0] Specify Right Hand part number
#   -lpart                [1] Specify Left Hand part number
#   --rstaff              [auto] Right-hand staff number for single-part MusicXML
#   --lstaff              [auto] Left-hand staff number for single-part MusicXML
#   --auto-routing        Resolve part/staff routing automatically (default)
#   --manual-routing      Use explicit -rpart/-lpart/--rstaff/--lstaff values
#   --quiet               Switch off verbosity
#   -m, --musescore       Open output in musescore after processing
#   -b, --below-beam      Show fingering numbers below beam line
#   -v, --with-vedo       Play 3D scene after processing
#   -z, --sound-off       Disable sound
#   -l, --left-only       Fingering for left hand only
#   -r, --right-only      Fingering for right hand only
#   --hand-size           Hand size preset [XXS, XS, S, M, L, XL, XXL]
#   --chord-note-stagger-s [0.05] Small note staggering used to represent chords
```

Routing defaults are automatic and shown in the run summary:
- In **2-part** piano scores, hands are selected by `-rpart/-lpart`.
- In **1-part, 2-staff** piano scores, hands are selected by staff (`RH -> staff 1`, `LH -> staff 2`).
- Use `--manual-routing` plus `--rstaff/--lstaff` to override routing explicitly.

### GUI Usage
Run `pianoplayer` with no filename to open the GUI, then:

![newgui](https://github.com/user-attachments/assets/d65a2fdb-2efd-4b5b-98e6-ba1ad1328dca)

- press **Import Score** (valid formats: *MusicXML/MXL, MuseSsore, MIDI, [PIG](http://beam.kisarazu.ac.jp/~saito/research/PianoFingeringDataset/)*)
- press **GENERATE** (`output.xml` is written)
- press **Musescore** to visualize the annotated score (Linux/macOS only)
- press **Quit** (or `q` / `Ctrl+W`) to close the GUI
- In **Advanced**, keep **Auto hand routing** enabled for default behavior, or disable it to set
  right/left part and staff manually.


#### Example output, as displayed in Musescore:

(If fingering numbers are not visible enough try with `-b` option.)


![bachinv4](https://user-images.githubusercontent.com/32848391/63605352-10f5f480-c5ce-11e9-8b00-34f1adc2e79b.png)


![pianoplayer3d](https://user-images.githubusercontent.com/32848391/63605322-0176ab80-c5ce-11e9-8213-b572d0303523.gif)


## How the algorithm works:
The algorithm minimizes the fingers speed needed to play a sequence of notes or chords by searching
through feasible combinations of fingerings.

One possible advantage of this algorithm over similar ones is that it is completely *dynamic*,
which means that it
takes into account the physical position and speed of fingers while moving on the keyboard
and the duration of each played note.
It is *not* based on a static look-up table of likely or unlikely combinations of fingerings.

Fingering a piano score can vary a lot from individual to individual, therefore there is not such
a thing as a "best" choice for fingering.
This algorithm is meant to suggest a fingering combination which is "optimal" in the sense that it
minimizes the effort of the hand avoiding unnecessary movements.

## Parameters you can change:
- Your hand size (from 'XXS' to 'XXL') which sets the relaxed distance between thumb and pinkie.
- The default part routing is `rpart=0` and `lpart=1`.
You can change it with `-rpart` and `-lpart` command line options.
- In single-part MusicXML files that contain 2 staves, routing defaults to `RH=staff 1` and
`LH=staff 2`. Use `--rstaff` and `--lstaff` to override this behavior.
- Depth of combinatorial search, from 5 up to 9 notes ahead of the currently playing note. By
default the algorithm selects this number automatically based on the duration of the notes to be played.

## Limitations
- Some specific fingering combinations, considered too unlikely in the first place, are excluded from the search (e.g. the 3rd finger crossing the 4th).
- Hands are always assumed independent from each other.
- In the 3D representation with sounds enabled, notes are played one after the other (no chords),
so the tempo within the measure is not always respected.
- Small notes/ornaments are ignored.
