|
|
|
/**
|
|
|
|
*
|
|
|
|
* Copyright (c) 2013 Pascal Gauthier.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be
|
|
|
|
* useful, but WITHOUT ANY WARRANTY; without even the implied
|
|
|
|
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public
|
|
|
|
* License along with this program; if not, write to the Free
|
|
|
|
* Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "PluginProcessor.h"
|
|
|
|
#include "PluginEditor.h"
|
|
|
|
#include "GlobalEditor.h"
|
|
|
|
#include "math.h"
|
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
using namespace ::std;
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
DexedAudioProcessorEditor::DexedAudioProcessorEditor (DexedAudioProcessor* ownerFilter)
|
|
|
|
: AudioProcessorEditor (ownerFilter),
|
|
|
|
midiKeyboard (ownerFilter->keyboardState, MidiKeyboardComponent::horizontalKeyboard)
|
|
|
|
{
|
|
|
|
LookAndFeel::setDefaultLookAndFeel(&dx_lnf);
|
|
|
|
|
|
|
|
setSize (865, 420);
|
|
|
|
|
|
|
|
processor = ownerFilter;
|
|
|
|
|
|
|
|
addAndMakeVisible(&cartridges);
|
|
|
|
cartridges.setEditableText(false);
|
|
|
|
cartridges.setJustificationType(Justification::centredLeft);
|
|
|
|
cartridges.setTextWhenNothingSelected(String::empty);
|
|
|
|
cartridges.setBounds(5, 5, 160, 18);
|
|
|
|
cartridges.addItemList(processor->cartManager.cartNames, 1);
|
|
|
|
cartridges.setSelectedItemIndex(0);
|
|
|
|
cartridges.addListener(this);
|
|
|
|
|
|
|
|
addAndMakeVisible (loadButton = new TextButton("LOAD"));
|
|
|
|
loadButton->setButtonText ("LOAD");
|
|
|
|
loadButton->addListener (this);
|
|
|
|
loadButton->setBounds(169, 5, 50, 18);
|
|
|
|
|
|
|
|
addAndMakeVisible(saveButton = new TextButton("SAVE"));
|
|
|
|
saveButton->setButtonText ("SAVE");
|
|
|
|
saveButton->addListener (this);
|
|
|
|
saveButton->setBounds (222, 5, 50, 18);
|
|
|
|
|
|
|
|
addAndMakeVisible (&programs);
|
|
|
|
programs.setEditableText (false);
|
|
|
|
programs.setJustificationType (Justification::centredLeft);
|
|
|
|
programs.setTextWhenNothingSelected (String::empty);
|
|
|
|
programs.setBounds(276, 5, 160, 18);
|
|
|
|
rebuildProgramCombobox();
|
|
|
|
programs.addListener(this);
|
|
|
|
|
|
|
|
addAndMakeVisible(storeButton = new TextButton("STORE"));
|
|
|
|
storeButton->setButtonText ("STORE");
|
|
|
|
storeButton->addListener (this);
|
|
|
|
storeButton->setBounds (439, 5, 50, 18);
|
|
|
|
|
|
|
|
addAndMakeVisible(aboutButton = new TextButton("ABOUT"));
|
|
|
|
aboutButton->setButtonText ("ABOUT");
|
|
|
|
aboutButton->addListener (this);
|
|
|
|
aboutButton->setBounds (805, 5, 50, 18);
|
|
|
|
|
|
|
|
|
|
|
|
// OPERATORS
|
|
|
|
addAndMakeVisible(&(operators[0]));
|
|
|
|
operators[0].setBounds(5, 39, 280, 90);
|
|
|
|
operators[0].bind(processor, 0);
|
|
|
|
|
|
|
|
addAndMakeVisible(&(operators[1]));
|
|
|
|
operators[1].setBounds(290, 39, 280, 90);
|
|
|
|
operators[1].bind(processor, 1);
|
|
|
|
|
|
|
|
addAndMakeVisible(&(operators[2]));
|
|
|
|
operators[2].setBounds(575, 39, 280, 90);
|
|
|
|
operators[2].bind(processor, 2);
|
|
|
|
|
|
|
|
addAndMakeVisible(&(operators[3]));
|
|
|
|
operators[3].setBounds(5, 133, 280, 90);
|
|
|
|
operators[3].bind(processor, 3);
|
|
|
|
|
|
|
|
addAndMakeVisible(&(operators[4]));
|
|
|
|
operators[4].setBounds(290, 133, 280, 90);
|
|
|
|
operators[4].bind(processor, 4);
|
|
|
|
|
|
|
|
addAndMakeVisible(&(operators[5]));
|
|
|
|
operators[5].setBounds(575, 133, 280, 90);
|
|
|
|
operators[5].bind(processor, 5);
|
|
|
|
|
|
|
|
// add the midi keyboard component..
|
|
|
|
addAndMakeVisible (&midiKeyboard);
|
|
|
|
|
|
|
|
// The DX7 is a badass on the bass, keep it that way
|
|
|
|
midiKeyboard.setLowestVisibleKey(24);
|
|
|
|
|
|
|
|
const int keyboardHeight = 90;
|
|
|
|
midiKeyboard.setBounds (4, getHeight() - keyboardHeight - 4, getWidth() - 8, keyboardHeight);
|
|
|
|
|
|
|
|
addAndMakeVisible(&global);
|
|
|
|
global.setBounds(5,235,855,90);
|
|
|
|
global.bind(processor);
|
|
|
|
|
|
|
|
updateUI();
|
|
|
|
startTimer(100);
|
|
|
|
}
|
|
|
|
|
|
|
|
DexedAudioProcessorEditor::~DexedAudioProcessorEditor() {
|
|
|
|
stopTimer();
|
|
|
|
processor->unbindUI();
|
|
|
|
}
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
void DexedAudioProcessorEditor::paint (Graphics& g) {
|
|
|
|
g.setColour(Colour(0xFF744420));
|
|
|
|
g.fillRoundedRectangle(0.0f, 0.0f, (float) getWidth(), (float) getHeight(), 0);
|
|
|
|
|
|
|
|
g.setColour(Colour(0xFFA87B67));
|
|
|
|
g.fillRoundedRectangle(0.0f, 30, (float) getWidth(), 200, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DexedAudioProcessorEditor::buttonClicked(Button *buttonThatWasClicked) {
|
|
|
|
|
|
|
|
if (buttonThatWasClicked == loadButton) {
|
|
|
|
FileChooser fc ("Import original DX sysex...", File::nonexistent, "*.syx;*.SYX;*.*", 1);
|
|
|
|
|
|
|
|
if ( fc.browseForFileToOpen()) {
|
|
|
|
String f = fc.getResults().getReference(0).getFullPathName();
|
|
|
|
uint8_t syx_data[4104];
|
|
|
|
ifstream fp_in(f.toRawUTF8(), ios::binary);
|
|
|
|
if (fp_in.fail()) {
|
|
|
|
AlertWindow::showMessageBoxAsync (AlertWindow::WarningIcon,
|
|
|
|
"Error",
|
|
|
|
"Unable to open: " + f);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
fp_in.read((char *)syx_data, 4104);
|
|
|
|
fp_in.close();
|
|
|
|
processor->importSysex((char *) &syx_data);
|
|
|
|
processor->setCurrentProgram(0);
|
|
|
|
rebuildProgramCombobox();
|
|
|
|
programs.setSelectedId(processor->getCurrentProgram()+1, NotificationType::dontSendNotification);
|
|
|
|
processor->updateHostDisplay();
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buttonThatWasClicked == saveButton) {
|
|
|
|
FileChooser fc ("Export DX sysex...", File::nonexistent, "*.syx", 1);
|
|
|
|
if ( fc.browseForFileToSave(true) ) {
|
|
|
|
String f = fc.getResults().getReference(0).getFullPathName();
|
|
|
|
char syx_data[4104];
|
|
|
|
|
|
|
|
exportSysex((char *) syx_data, (char *) &processor->sysex);
|
|
|
|
|
|
|
|
ofstream fp_out(f.toRawUTF8(), ios::binary);
|
|
|
|
fp_out.write((char *)syx_data, 4104);
|
|
|
|
fp_out.close();
|
|
|
|
|
|
|
|
if (fp_out.fail()) {
|
|
|
|
AlertWindow::showMessageBoxAsync (AlertWindow::WarningIcon,
|
|
|
|
"Error",
|
|
|
|
"Unable to write: " + f);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buttonThatWasClicked == storeButton) {
|
|
|
|
storeProgram();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
if (buttonThatWasClicked == cartButton) {
|
|
|
|
AlertWindow dialog(String("Builtin cartridges"), "", AlertWindow::NoIcon, this);
|
|
|
|
dialog.addComboBox(String("cart"), processor->cartManager.cartNames);
|
|
|
|
dialog.addButton("OK", 0, KeyPress(KeyPress::returnKey));
|
|
|
|
dialog.addButton("Cancel", 1, KeyPress(KeyPress::escapeKey));
|
|
|
|
if ( dialog.runModalLoop() == 0 ) {
|
|
|
|
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
if (buttonThatWasClicked == aboutButton) {
|
|
|
|
AlertWindow::showMessageBoxAsync(AlertWindow::NoIcon, "DEXED - DX Emulator 0.3", "https://github.com/asb2m10/dexed\n"
|
|
|
|
"(c) 2013-2014 Pascal Gauthier\nUnder the GPL v2\n\n"
|
|
|
|
"Based on Music Synthesizer for Android\nhttps://code.google.com/p/music-synthesizer-for-android");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
AlertWindow::showMessageBoxAsync(AlertWindow::WarningIcon, "Sorry", "Soon !");
|
|
|
|
}
|
|
|
|
|
|
|
|
void DexedAudioProcessorEditor::comboBoxChanged (ComboBox* comboBoxThatHasChanged) {
|
|
|
|
if ( comboBoxThatHasChanged == &programs ) {
|
|
|
|
processor->setCurrentProgram(programs.getSelectedId()-1);
|
|
|
|
processor->updateHostDisplay();
|
|
|
|
} else {
|
|
|
|
int idx = comboBoxThatHasChanged->getSelectedItemIndex();
|
|
|
|
processor->loadBuiltin(idx);
|
|
|
|
processor->setCurrentProgram(0);
|
|
|
|
rebuildProgramCombobox();
|
|
|
|
programs.setSelectedId(processor->getCurrentProgram()+1, NotificationType::dontSendNotification);
|
|
|
|
processor->updateHostDisplay();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DexedAudioProcessorEditor::timerCallback() {
|
|
|
|
int32_t env[6];
|
|
|
|
|
|
|
|
if ( processor->peekEnvStatus(env) == false )
|
|
|
|
return;
|
|
|
|
|
|
|
|
for(int i=0;i<6;i++) {
|
|
|
|
operators[i].updateGain(sqrt(env[5 - i]) / 8196);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DexedAudioProcessorEditor::updateUI() {
|
|
|
|
for(int i=0;i<processor->ctrl.size();i++) {
|
|
|
|
processor->ctrl[i]->updateComponent();
|
|
|
|
}
|
|
|
|
for(int i=0;i<6;i++) {
|
|
|
|
operators[i].updateDisplay();
|
|
|
|
}
|
|
|
|
|
|
|
|
int id = processor->getCurrentProgram() + 1;
|
|
|
|
programs.setSelectedId(id, NotificationType::dontSendNotification);
|
|
|
|
|
|
|
|
global.updateDisplay();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DexedAudioProcessorEditor::rebuildProgramCombobox() {
|
|
|
|
programs.clear(NotificationType::dontSendNotification);
|
|
|
|
for(int i=0;i<processor->getNumPrograms();i++) {
|
|
|
|
String id;
|
|
|
|
id << (i+1) << ". " << processor->getProgramName(i);
|
|
|
|
programs.addItem(id, i+1);
|
|
|
|
}
|
|
|
|
programs.setSelectedId(processor->getCurrentProgram()+1, NotificationType::dontSendNotification);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DexedAudioProcessorEditor::storeProgram() {
|
|
|
|
String currentName(processor->getProgramName(processor->getCurrentProgram()));
|
|
|
|
char destSysex[4096];
|
|
|
|
File *externalFile = NULL;
|
|
|
|
|
|
|
|
memcpy(&destSysex, processor->sysex, 4096);
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
String msg;
|
|
|
|
if ( externalFile == NULL ) {
|
|
|
|
msg = "Store program to current cartridge";
|
|
|
|
} else {
|
|
|
|
msg = "Store program to " + externalFile->getFileName();
|
|
|
|
}
|
|
|
|
|
|
|
|
AlertWindow dialog(String("Store Program"), msg, AlertWindow::NoIcon, this);
|
|
|
|
dialog.addTextEditor(String("Name"), currentName, String("Name"), false);
|
|
|
|
// TODO: fix the name length to 10
|
|
|
|
|
|
|
|
StringArray programs;
|
|
|
|
extractProgramNames((char *) &destSysex, programs);
|
|
|
|
|
|
|
|
dialog.addComboBox(String("Dest"), programs);
|
|
|
|
dialog.addButton("OK", 0, KeyPress(KeyPress::returnKey));
|
|
|
|
dialog.addButton("Cancel", 1, KeyPress(KeyPress::escapeKey));
|
|
|
|
dialog.addButton("External File", 2, KeyPress());
|
|
|
|
int response = dialog.runModalLoop();
|
|
|
|
|
|
|
|
if ( response == 2 ) {
|
|
|
|
FileChooser fc("Destination Sysex", File::nonexistent, "*.syx;*.SYX;*.*", 1);
|
|
|
|
|
|
|
|
if ( fc.browseForFileToOpen() ) {
|
|
|
|
if ( externalFile != NULL )
|
|
|
|
delete externalFile;
|
|
|
|
|
|
|
|
MemoryBlock block;
|
|
|
|
externalFile = new File(fc.getResults().getReference(0));
|
|
|
|
if ( externalFile->loadFileAsData(block) ) {
|
|
|
|
block.copyTo(destSysex, 6, 4096);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
AlertWindow::showMessageBoxAsync(AlertWindow::WarningIcon, "Read error", "Unable to read file");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( response == 0 ) {
|
|
|
|
TextEditor *name = dialog.getTextEditor(String("Name"));
|
|
|
|
ComboBox *dest = dialog.getComboBoxComponent(String("Dest"));
|
|
|
|
|
|
|
|
int programNum = dest->getSelectedItemIndex();
|
|
|
|
String programName = name->getText();
|
|
|
|
if ( programName.length() > 10 ) {
|
|
|
|
int toStrip = programName.length() - 10;
|
|
|
|
programName.dropLastCharacters(toStrip);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( externalFile == NULL ) {
|
|
|
|
packProgram((uint8_t *) processor->sysex, (uint8_t *) processor->data, programNum, programName);
|
|
|
|
processor->programNames.set(programNum, programName);
|
|
|
|
rebuildProgramCombobox();
|
|
|
|
processor->setCurrentProgram(programNum);
|
|
|
|
processor->updateHostDisplay();
|
|
|
|
} else {
|
|
|
|
packProgram((uint8_t *) &destSysex, (uint8_t *) processor->data, programNum, programName);
|
|
|
|
char sysexFile[4104];
|
|
|
|
exportSysex((char *) &sysexFile, (char *) &destSysex);
|
|
|
|
if ( ! externalFile->replaceWithData(sysexFile, 4104) ) {
|
|
|
|
AlertWindow::showMessageBoxAsync(AlertWindow::WarningIcon, "Write error", "Unable to write file");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( externalFile != NULL )
|
|
|
|
delete externalFile;
|
|
|
|
}
|