npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

blob-opera-midi

v1.0.11

Published

Convert MIDI arrangements to Blob Opera JSON.

Downloads

41

Readme

blob-opera-midi

Quick and dirty tool to convert 4-part MIDI arrangments to Blob Opera format JSON.

Background

Blob Opera is a "machine learning experiment by David Li in collaboration with Google Arts & Culture", allowing users to "[create their] own opera inspired song with Blob Opera - no music skills required".

It's definitely worth playing with the Blob Opera before using this tool. The musical toy can be controlled by dragging the blobs with your mouse or, if you have one connected, a MIDI input device like a keyboard or sequencer. Both of these methods have limitations. Mouse control is imprecise and only affects one voice at a time with no manual harmony. MIDI control can be used to play the blobs like an organ but cannot control individual voices as a range of absolute MIDI pitches is assigned to each blob, making complex arrangements difficult. To enable the blobs to sing arbitrary choral (SATB) arrangements I wrote a tool that converts multitrack MIDI files into the file format used by the blobs to play included example songs, and found a method to cause the blobs to load my file instead of the expected example file.

Here are some of my results (Twitter video links):

Installation

$ npm install -g blob-opera-midi

Usage

Converting the MIDI file

If your MIDI file is already exactly 4 tracks in SATB order:

  • $ blob-opera-midi song.mid
  • JSON file will be output to <filename>.mid.json

If your MIDI is not in track order or you want to preview the track assignments:

  • $ blob-opera-midi song.mid -i or $ blob-opera-midi song.mid --interactive
  • Arrow keys to select parts to map, enter to bring up a dialog with a number input corresponding to MIDI tracks from track graph on right, click (yes, with your mouse) the export button, then esc or q to exit. Alternatively, you can use the ctrl-e hotkey to export and immediately exit.
  • JSON file will be exported as <filename>.mid.json.

Other command-line flags:

  • -r or --random to add slight timing drift (may provide more naturalistic sound). By default, no drift is added.
  • -f or --free-pitch to allow notes outside of the comfortable range of the blobs. By default, pitches are clamped between 48 and 70, although the actual note produced by a blob may be in a different octave depending on range.
  • -c or --christmas to make the blobs wear santa hats (no effect when using Method 2 to load a song).

Sideloading the JSON file

Method 1: Recording

  1. Open the Blob Opera web page, preferably with Google Chrome.
  2. Open the developer tools; usually by right-clicking anywhere and choosing Inspect...
  3. Browse to the application source file (app.js) and add a breakpoint on the first line of this function:
    t.prototype.finishRecording = function() {
               if (!this.isRecording) // ADD THE BREAKPOINT HERE
                   return null;
               var t = this.currentRecording;
               return function(t, e) {
                   for (var n = 1 / 0, i = 0, r = t.parts; i < r.length; i++) {
                       (l = r[i]).notes.length > 0 && (n = Math.min(n, l.notes[0].timeSeconds))
                   }
                   for (var o = e - n, a = 0, s = t.parts; a < s.length; a++)
                       for (var l, u = 0, c = (l = s[a]).notes; u < c.length; u++) {
                           c[u].timeSeconds += o
                       }
               }(t, .2),
               this.isRecording = !1,
               this.currentRecording = null,
               t
           }
  4. Start a new recording by pressing the red button on the bottom-left corner and stop it again.
  5. Now you've hit the breakpoint; type this.currentRecording = <contents> where <contents> should be replaced by the contents of the JSON file generated previously.
  6. Resume the execution by clicking the play button on the debugger.
  7. Wait a few seconds for the speed to stabilize and stop it.

Method 2 (deprecated): Load as holiday song

  1. Follow steps 1-3 above, but instead set a breakpoint at 9939 (e.opera.enterPlaying(n))
  2. Click the holiday icon and select a song
  3. Breakpoint should trigger. Type n = <contents> where <contents> is the contents of your JSON file.
  4. Resume the execution by clicking the play button on the debugger.
  5. Immediately click stop and wait for the speed to sync, then click play.

You can also save songs sideloaded with this method to a shareable URL by encoding it using te.RecordingMessage.encode(<JSON>).finish() at line 9428 of the main app file:

Screenshot of Chrome dev tools with a breakpoint set on line 9428 of prettified app source

Objections

But isn't this really hacky?

Yes.

Can't you just control the Blob Opera directly with MIDI signals?

Yes, but that's not interesting to me and this has advantages over that like temporally consistent vocalizations without having to do any manual programming.

There's something wrong with the timing!

Probably. Submit a github issue!

Related links