Spaces:
Runtime error
Runtime error
import matplotlib.pyplot as plt | |
import gradio as gr | |
from load import LoadModel | |
from generate import GenerateMidiText | |
from constants import INSTRUMENT_CLASSES, INSTRUMENT_TRANSFER_CLASSES | |
from decoder import TextDecoder | |
from utils import get_miditok, index_has_substring | |
from playback import get_music | |
from matplotlib import pylab | |
import sys | |
import matplotlib | |
from generation_utils import plot_piano_roll | |
import numpy as np | |
matplotlib.use("Agg") | |
sys.modules["pylab"] = pylab | |
model_repo = "JammyMachina/elec-gmusic-familized-model-13-12__17-35-53" | |
n_bar_generated = 8 | |
# model_repo = "JammyMachina/improved_4bars-mdl" | |
# n_bar_generated = 4 | |
model, tokenizer = LoadModel( | |
model_repo, | |
from_huggingface=True, | |
).load_model_and_tokenizer() | |
miditok = get_miditok() | |
decoder = TextDecoder(miditok) | |
def define_prompt(state, genesis): | |
if len(state) == 0: | |
input_prompt = "PIECE_START " | |
else: | |
input_prompt = genesis.get_whole_piece_from_bar_dict() | |
return input_prompt | |
def generator( | |
label, | |
regenerate, | |
temp, | |
density, | |
instrument, | |
state, | |
piece_by_track, | |
add_bars=False, | |
add_bar_count=1, | |
): | |
genesis = GenerateMidiText(model, tokenizer, piece_by_track) | |
track = {"label": label} | |
inst = next( | |
( | |
inst | |
for inst in INSTRUMENT_TRANSFER_CLASSES | |
if inst["transfer_to"] == instrument | |
), | |
{"family_number": "DRUMS"}, | |
)["family_number"] | |
inst_index = -1 # default to last generated | |
if state != []: | |
for index, instrum in enumerate(state): | |
if instrum["label"] == track["label"]: | |
inst_index = index # changing if exists | |
# Generate | |
if not add_bars: | |
# Regenerate | |
if regenerate: | |
state.pop(inst_index) | |
genesis.delete_one_track(inst_index) | |
generated_text = ( | |
genesis.get_whole_piece_from_bar_dict() | |
) # maybe not useful here | |
inst_index = -1 # reset to last generated | |
# NEW TRACK | |
input_prompt = define_prompt(state, genesis) | |
generated_text = genesis.generate_one_new_track( | |
inst, density, temp, input_prompt=input_prompt | |
) | |
regenerate = True # set generate to true | |
else: | |
# NEW BARS | |
genesis.generate_n_more_bars(add_bar_count) # for all instruments | |
generated_text = genesis.get_whole_piece_from_bar_dict() | |
decoder.get_midi(generated_text, "mixed.mid") | |
mixed_inst_midi, mixed_audio = get_music("mixed.mid") | |
inst_text = genesis.get_selected_track_as_text(inst_index) | |
inst_midi_name = f"{instrument}.mid" | |
decoder.get_midi(inst_text, inst_midi_name) | |
_, inst_audio = get_music(inst_midi_name) | |
piano_roll = plot_piano_roll(mixed_inst_midi) | |
track["text"] = inst_text | |
state.append(track) | |
return ( | |
inst_text, | |
(44100, inst_audio), | |
piano_roll, | |
state, | |
(44100, mixed_audio), | |
regenerate, | |
genesis.piece_by_track, | |
) | |
def instrument_row(default_inst, row_id): | |
with gr.Row(): | |
row = gr.Variable(row_id) | |
with gr.Column(scale=1, min_width=100): | |
inst = gr.Dropdown( | |
sorted([inst["transfer_to"] for inst in INSTRUMENT_TRANSFER_CLASSES]) | |
+ ["Drums"], | |
value=default_inst, | |
label="Instrument", | |
) | |
temp = gr.Dropdown( | |
[0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9, 1], | |
value=0.7, | |
label="Creativity", | |
) | |
density = gr.Dropdown([1, 2, 3], value=3, label="Note Density") | |
with gr.Column(scale=3): | |
output_txt = gr.Textbox( | |
label="output", lines=10, max_lines=10, show_label=False | |
) | |
with gr.Column(scale=1, min_width=100): | |
inst_audio = gr.Audio(label="TRACK Audio", show_label=True) | |
regenerate = gr.Checkbox(value=False, label="Regenerate", visible=False) | |
# add_bars = gr.Checkbox(value=False, label="Add Bars") | |
# add_bar_count = gr.Dropdown([1, 2, 4, 8], value=1, label="Add Bars") | |
gen_btn = gr.Button("Generate") | |
gen_btn.click( | |
fn=generator, | |
inputs=[row, regenerate, temp, density, inst, state, piece_by_track], | |
outputs=[ | |
output_txt, | |
inst_audio, | |
piano_roll, | |
state, | |
mixed_audio, | |
regenerate, | |
piece_by_track, | |
], | |
) | |
with gr.Blocks() as demo: | |
piece_by_track = gr.State([]) | |
state = gr.State([]) | |
title = gr.Markdown( | |
""" # Demo-App of The-Jam-Machine | |
A Generative AI trained on text transcription of MIDI music """ | |
) | |
track1_md = gr.Markdown(""" ## Mixed Audio and Piano Roll """) | |
mixed_audio = gr.Audio(label="Mixed Audio") | |
piano_roll = gr.Plot(label="Piano Roll", show_label=False) | |
description = gr.Markdown( | |
""" | |
For each **TRACK**, choose your **instrument** along with **creativity** (temperature) and **note density**. Then, hit the **Generate** Button! | |
You can have a look at the generated text; but most importantly, check the **piano roll** and listen to the TRACK audio! | |
If you don't like the track, hit the generate button to regenerate it! Generate more tracks and listen to the **mixed audio**! | |
""" | |
) | |
track1_md = gr.Markdown(""" ## TRACK 1 """) | |
instrument_row("Drums", 0) | |
track1_md = gr.Markdown(""" ## TRACK 2 """) | |
instrument_row("Synth Bass 1", 1) | |
track1_md = gr.Markdown(""" ## TRACK 2 """) | |
instrument_row("Synth Lead Square", 2) | |
# instrument_row("Piano") | |
demo.launch(debug=True) | |
""" | |
TODO: reset button | |
TODO: add a button to save the generated midi | |
TODO: add improvise button | |
TODO: set values for temperature as it is done for density | |
TODO: Add bar should be now set for the whole piece - regenerrate should regenerate the added bars only on all instruments | |
TODO: row height to fix | |
TODO: reset state of tick boxes after used maybe (regenerate, add bars) ; | |
TODO: block regenerate if add bar on | |
""" | |