Plugins for 3.x
Not to be confused with Musescore 4's VST and VSTi feature. Other plugin versions see Plugin development.
Introduction
Edit this page to share Musescore 3 plugin programming knowledge. Discussion thread.
Writing and running plugins requires,
- Musescore Studio installed,
- Javascript knowledge, and
- QML knowledge.
QML
Written with just enough jargon inaccurately, like "the sun rises in the east"..
Qt company invented QML,
- rules to write .qml files, including
- extended rules to write .qml files, by adding
import
statements- Qt + (module name) eg Qt.labs.settings
- Qt Quick
- Data
- UI, the fancier buttons etc
- Qt Quick + (module name) eg Qt Quick Controls, the even more fancier buttons etc
- rules to write C++ (.cpp, .h etc) files including separated copies share similar naming of some items above,
- Qt Widget, fine-control UI programming.
- QML Widget (Qt Quick Widget), abstract UI programming.
Note that item 3 is for Musescore main app devs only. Things with similar names use different rules, don't read the wrong doc.
//https://doc.qt.io/archives/qt-5.9/qtqml-documents-topic.html import QtQuick 2.0 Rectangle { width: 100; height: 100 color: "red" MouseArea { anchors.fill: parent onClicked: console.log("Button clicked!") } }
Any of these work,
onClicked: console.log("Button clicked!") onClicked: function(){ console.log("Button clicked!") }
Use javascript / ECMAScript 7th edition function. Unsupported are arrow function, spread syntax, array.fill etc.
Musescore plugin qml
Musescore plugin qml uses,
- all rules of .qml files,
- not available is Qt Quick Controls style (import QtQuick.Controls.Material). See https://musescore.org/en/node/21875#comment-98406;
- Qt Webengine only available on Windows;
- functions made available by the Musescore dev team.
Capability and limitation
This non-exclusive list contains observations reported by coders.
- Examples of basic functionality
- Change a note's property such as pitch, color
- Add or remove score elements such as notes, rests, symbol, text
- Read or save tags in File > Score Properties
- Play the audio of a score or part of score
- Create a new staff (Cannot change its name afterwards)
- Create/update text and symbols eg dynamics, lyrics etc
- Create/update tempo
- Change playback speed (cmd('playback-speed-increase'))
- Mute or change volume of an instrument
- Create/update MIDI Play Event
- Conversion of a different score format to and from MuseScore
- Send and receive score data through internet
- Opens a local or online webpage inside MuseScore
- Save or load a score or any file
- Start another program on your Windows/iOS outside MuseScore
- Seems unachievable (please share if you find workaround, for example a suitable cmd() command )
- Run plugin when audio is playing, or detect whether audio is playing, or realtime Playback Caret/cursor position.
- Directly changing the Playback Caret (Can read Editing Caret position)
- Change instrument name (Can read, but not write)
- Navigate into content, eg title text, inside a frame (user must select the text inside manually with a mouse click first) (Can get to the frame by cmd('prev-element'))
- Add spanners such as pedal marking to a score
- Create or update key signatures in a score
- Read clef info, see https://musescore.org/en/node/357788
- Send MIDI CC
- Mimic a click on a palette item
- Add plugin as a toolbar button
- Add plugin as an item in right click context menu
- Manipulation of MuseScore's window eg minimize
- Use a non build-into-Musescore text font, see workaround post (Can use with pre setup inside qml, see snippet)
- Expose the program with new hooks or API entry points, or create localhost server / services.
- Inserting image. See https://musescore.org/en/node/373264
See also https://musescore.org/en/node/372023
Alternatives to QML plugin
- Run Musescore with Command line
- Edit a score file (.mscx is plaintext):
- save as .mscz : rename as .zip, extract, edit, zip ; or
- save as .mscx and edit.
- elsewhere's parser https://musescore.org/en/project/musicxml-parser
- mirabilos' tools https://musescore.org/en/node/321691#comment-1082043
- yonah_ag's parser (retired) https://musescore.org/en/node/323083
- AutoHotkey Kit for MuseScore.
- Get data of realtime midi output or exported .midi.
- Use a non build-into-Musescore font with this workaround post.
- Install Musescore 3.x Nightly and use the (experimental) scripting/macro function to record and replay sequence of commands "Tools → Script Recorder".
Workflow and external editors
Edit plugin with,
- Plugin Creator, or
- external editor / IDE (Qt Creator, Sublime Text, Notepad++ (TIPS: use the Qt user-defined language), vim, emac, https://duckduckgo.com/?q=qml+online+editor etc).
Plugin scripts are cached at program startup. To run a script updated afterwards in external editor,
- Restart Musescore and then run from the plugin menu; or
- Load the script inside Plugin Creator
- Inside the Plugin Creator, open an existing .qml file
- Press run
- After subsequent edits in external editor, it is required to press Refresh in the Plugin Creator.
- It is not required to restart Musescore when using this method.
Code examples
Common tasks
Changing menu name, window behavior
Get elements
- Read Score elements, internal score structure.
- curScore.newCursor() to create a Cursor, transverse and find. For more info, open and study walk.qml
- curScore.newCursor() to create a Cursor, sync with current mouse selection example
- select with mouse click then get curScore.selection.elements[ ]
- listen to state.selectionChanged, get curScore.selection example
- select with curScore.selection.selectRange() , get curScore.selection example
Read meaningful info of element type
- Print
.name
property.
Match and compare element types
Match and compare reliably with .type
property eg Accidental.FLAT
. See enumeration / enum. These pages are useful,
Try log type enum string snippet and Makeshift REPL plugin function tester plugin for enum and keyword autocompletion.
Note starting time, ending time, duration (note value)
- api doc Tick Length values
- Cursor and segment has .tick, snippet
Note pitch
- use case and snippets
- api doc Note Pitch values
- api doc Tonal Pitch Class
Placement of objects, cursor track and tick
cursor.rewindToTick(x) will only work correctly if there is a segment at x. Otherwise it will put the cursor at the next available segment (in the current track) after x, see https://musescore.org/en/node/345409#comment-1206211
Save settings to score, tagging
Save settings to OS
eg registry in windows
File saving and loading
- api writeScore, readScore
- custom data in MuseScore several methods see this snippet and notes
- study the Daily Log plugin
- FileIO for OS level op, usage see Object Explorer plugin
Instrument / staff info eg instrument name, and channel audio status
Create/update MIDI Play Event
- API doc PlayEvent
- study the Arpeggiator plugin and the articulation and ornamentation control plugin
Fetching webpage
- snippet
- study the updated online service version of ABC Import plugin
- the Harmony analysis tool plugin
Starting another program, command line and parameters
Get UI theme status
Not engraving style
SystemPalette { }
. Source post. See color and metric details in source code.
Browsing help sites
QML help sites
doc.qt.io
When browsing https://doc.qt.io, note that
- "C++ classes" pages are not useful in qml.
- Check qt version on URL. Musescore 3.6.2 requires Qt 5.9.9 syntax. Some pages can be auto redirected by changing "6" in URL to "5.9". eg changing
https://doc.qt.io/qt-6/qml-qtquick-wheelevent.html
tohttps://doc.qt.io/qt-5.9/qml-qtquick-wheelevent.html
and refreshing jumps tohttps://doc.qt.io/archives/qt-5.9/qml-qtquick-wheelevent.html
. -
Use a matching the import module version number on pages returned from site search. eg site search "button" returns QtQuick 1 Button but not QtQuick 2 Button.
musescore.org
plugin help sites
musescore.github.io
- Official Musescore Plugin API Documentation. Try the search on the top right if menus are hard to understand.
- List of all API object types
The cmd ( ) method
Unstable. Backup QML. Use cmd() to run internal program actions.
- complete list
- complete list in CSV format
- calculuswhiz's handpicked list
- This command parser plugin makes them more scriptable
musescore.org
- Must read about The Plugin Base Component Musescore { }.
- Notes, use case and snippets.
Useful Resource
The Plugin Base Component Musescore { }
- Api doc
- Component.onCompleted
Run once, right after the plugin component is created, before console.log() is functional - You should not use Qt.quit() in your plugins, see this post on why, and possible different approaches. Qt.quit() crashes Musescore 4.0 plugins.
- Set pluginType: "dock" for normal behavior eg stay on top; "dialog" is quite outdated.
- Create plugin as an action instead of a persistent object with prop and methods, because when the user close and reopen the plugin window, it may cause unwanted behavior.
- This post explains the nature of onRun() and Qt.quit(). The code Qt.quit() may not work the way you expect.
- The plugin window onClosing Event is not accessable from QML. The [Plugin API base] is in a QQuickView in a QWidget window docked inside a QDockWidget , see code.
- No way to detect plugin's open or closed.
- After plugin window is closed, listeners will continue to run, eg experimental onScoreStateChanged Handler
- Workaround palette-like plugin
- use Qt Labs Settings to save data across sessions
- other tips also check out the snippets and notes page
Score elements, internal score structure
- Musescore Internal Score Representation
- Confused by API Class Inheritance Hierarchy? Study inheritance object oriented programming.
- Use the debugger by doing the following steps
- Install MuseScore Nightly
- Append -e in shortcut and start
- Edit > Debugger
- Due to Musescore dev team's plugin system design, all element types contains all properties (keys) and useless properties have
undefined
values. There is no decent way to filter out valid properties except,- Trying out to see if code executed works,
- Digging into the source code,
- Try Object Explorer plugin which exhaustively print out element info, also see this well documented code,
- Try Makeshift REPL plugin function tester plugin's "Diff an element" example.
Bundled resource (qrc protocol)
- Icons, symbols and fonts (not soundfont) see this snippet
- musescore.qrc (src ref)
Searchable museScore 3.6.2 source code
- https://github.com/fp22june/MuseScoreTag362
/libmscore
- API 3.5 (for Musescore 3.6)
/mscore/plugin/api
- Some files are mirrored on the doxygen website.
Translation
Troubleshooting with help from community
- Identify the problem. If it is QML or javascript, also seek help on stackoverflow and forum.qt.io.
- Use google and the musescore.org forum search before posting new questions.
- musescore.org plugins forum,
- Try to provide a Minimal, Reproducible Example, the community is usually more than happy to help.
- Enclose qml codes between
<qml>
and</qml>
to get better layout and to avoid musescore.org auto escaping eg to show & instead of&
Share your plugin
- Posted plugin project on musescore.org. Fill out the API compatibility 3.x or else your plugin with remain hidden to most users.
- Chat on the plugin forum.
Licensing
- Writing in QML language is unlikely to be the ground on which plugin script subjects to Qt licensing.
- Opening in Musescore is unlikely to be the ground on which plugin script subjects to MuseScore 3 GPL licensing.