mirror of
https://github.com/juce-framework/JUCE.git
synced 2026-01-10 23:44:24 +00:00
541 lines
19 KiB
C++
541 lines
19 KiB
C++
/*
|
|
==============================================================================
|
|
|
|
This file is part of the JUCE library.
|
|
Copyright (c) 2013 - Raw Material Software Ltd.
|
|
|
|
Permission is granted to use this software under the terms of either:
|
|
a) the GPL v2 (or any later version)
|
|
b) the Affero GPL v3
|
|
|
|
Details of these licenses can be found at: www.gnu.org/licenses
|
|
|
|
JUCE 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.
|
|
|
|
------------------------------------------------------------------------------
|
|
|
|
To release a closed-source product which uses JUCE, commercial licenses are
|
|
available: visit www.juce.com for more information.
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
|
|
class ModulesPanel : public Component,
|
|
private TableListBoxModel,
|
|
private ValueTree::Listener,
|
|
private Button::Listener
|
|
{
|
|
public:
|
|
ModulesPanel (Project& p)
|
|
: project (p),
|
|
modulesValueTree (p.getModules().state),
|
|
addWebModuleButton ("Download and add a module..."),
|
|
updateModuleButton ("Install updates to modules..."),
|
|
setCopyModeButton ("Set copy-mode for all modules..."),
|
|
copyPathButton ("Set paths for all modules...")
|
|
{
|
|
table.getHeader().addColumn ("Module", nameCol, 180, 100, 400, TableHeaderComponent::notSortable);
|
|
table.getHeader().addColumn ("Installed Version", versionCol, 100, 100, 100, TableHeaderComponent::notSortable);
|
|
table.getHeader().addColumn ("Available Version", updateCol, 100, 100, 100, TableHeaderComponent::notSortable);
|
|
table.getHeader().addColumn ("Make Local Copy", copyCol, 100, 100, 100, TableHeaderComponent::notSortable);
|
|
table.getHeader().addColumn ("Paths", pathCol, 250, 100, 600, TableHeaderComponent::notSortable);
|
|
|
|
table.setModel (this);
|
|
table.setColour (TableListBox::backgroundColourId, Colours::transparentBlack);
|
|
addAndMakeVisible (table);
|
|
table.updateContent();
|
|
table.setRowHeight (20);
|
|
|
|
addAndMakeVisible (addWebModuleButton);
|
|
addAndMakeVisible (updateModuleButton);
|
|
addAndMakeVisible (setCopyModeButton);
|
|
addAndMakeVisible (copyPathButton);
|
|
addWebModuleButton.addListener (this);
|
|
updateModuleButton.addListener (this);
|
|
updateModuleButton.setEnabled (false);
|
|
setCopyModeButton.addListener (this);
|
|
setCopyModeButton.setTriggeredOnMouseDown (true);
|
|
copyPathButton.addListener (this);
|
|
copyPathButton.setTriggeredOnMouseDown (true);
|
|
|
|
modulesValueTree.addListener (this);
|
|
lookAndFeelChanged();
|
|
}
|
|
|
|
void paint (Graphics& g) override
|
|
{
|
|
if (webUpdateThread == nullptr)
|
|
webUpdateThread = new WebsiteUpdateFetchThread (*this);
|
|
|
|
IntrojucerLookAndFeel::fillWithBackgroundTexture (*this, g);
|
|
}
|
|
|
|
void resized() override
|
|
{
|
|
Rectangle<int> r (getLocalBounds().reduced (5, 4));
|
|
|
|
table.setBounds (r.removeFromTop (table.getRowPosition (getNumRows() - 1, true).getBottom() + 20));
|
|
|
|
Rectangle<int> buttonRow (r.removeFromTop (32).removeFromBottom (28));
|
|
addWebModuleButton.setBounds (buttonRow.removeFromLeft (jmin (260, r.getWidth() / 3)));
|
|
buttonRow.removeFromLeft (8);
|
|
updateModuleButton.setBounds (buttonRow.removeFromLeft (jmin (260, r.getWidth() / 3)));
|
|
buttonRow.removeFromLeft (8);
|
|
|
|
buttonRow = r.removeFromTop (34).removeFromBottom (28);
|
|
setCopyModeButton.setBounds (buttonRow.removeFromLeft (jmin (260, r.getWidth() / 3)));
|
|
buttonRow.removeFromLeft (8);
|
|
copyPathButton.setBounds (buttonRow.removeFromLeft (jmin (260, r.getWidth() / 3)));
|
|
}
|
|
|
|
int getNumRows() override
|
|
{
|
|
return project.getModules().getNumModules();
|
|
}
|
|
|
|
void paintRowBackground (Graphics& g, int /*rowNumber*/, int width, int height, bool rowIsSelected) override
|
|
{
|
|
g.setColour (rowIsSelected ? Colours::lightblue.withAlpha (0.4f)
|
|
: Colours::white.withAlpha (0.4f));
|
|
g.fillRect (0, 0, width, height - 1);
|
|
}
|
|
|
|
void paintCell (Graphics& g, int rowNumber, int columnId, int width, int height, bool /*rowIsSelected*/) override
|
|
{
|
|
String text;
|
|
const String moduleID (project.getModules().getModuleID (rowNumber));
|
|
|
|
if (columnId == nameCol)
|
|
{
|
|
text = moduleID;
|
|
}
|
|
else if (columnId == versionCol)
|
|
{
|
|
text = project.getModules().getModuleInfo (moduleID).getVersion();
|
|
|
|
if (text.isEmpty())
|
|
text = "?";
|
|
}
|
|
else if (columnId == updateCol)
|
|
{
|
|
if (listFromWebsite != nullptr)
|
|
{
|
|
if (const ModuleDescription* m = listFromWebsite->getModuleWithID (moduleID))
|
|
{
|
|
if (m->getVersion() != project.getModules().getModuleInfo (moduleID).getVersion())
|
|
text = m->getVersion() + " available";
|
|
else
|
|
text = "Up-to-date";
|
|
}
|
|
else
|
|
text = "?";
|
|
}
|
|
else
|
|
{
|
|
text = "-";
|
|
}
|
|
}
|
|
else if (columnId == copyCol)
|
|
{
|
|
text = project.getModules().shouldCopyModuleFilesLocally (moduleID).getValue()
|
|
? "Yes" : "No";
|
|
}
|
|
else if (columnId == pathCol)
|
|
{
|
|
StringArray paths;
|
|
|
|
for (Project::ExporterIterator exporter (project); exporter.next();)
|
|
paths.addIfNotAlreadyThere (exporter->getPathForModuleString (moduleID).trim());
|
|
|
|
text = paths.joinIntoString (", ");
|
|
}
|
|
|
|
g.setColour (Colours::black);
|
|
g.setFont (height * 0.65f);
|
|
g.drawText (text, Rectangle<int> (width, height).reduced (4, 0), Justification::centredLeft, true);
|
|
}
|
|
|
|
void cellDoubleClicked (int rowNumber, int, const MouseEvent&) override
|
|
{
|
|
const String moduleID (project.getModules().getModuleID (rowNumber));
|
|
|
|
if (moduleID.isNotEmpty())
|
|
if (ProjectContentComponent* pcc = findParentComponentOfClass<ProjectContentComponent>())
|
|
pcc->showModule (moduleID);
|
|
}
|
|
|
|
void deleteKeyPressed (int row) override
|
|
{
|
|
project.getModules().removeModule (project.getModules().getModuleID (row));
|
|
}
|
|
|
|
void webUpdateFinished (const ModuleList& newList)
|
|
{
|
|
listFromWebsite = new ModuleList (newList);
|
|
|
|
table.updateContent();
|
|
table.repaint();
|
|
|
|
updateModuleButton.setEnabled (getUpdatableModules().size() != 0);
|
|
}
|
|
|
|
void buttonClicked (Button* b)
|
|
{
|
|
if (b == &addWebModuleButton) showAddModuleMenu();
|
|
else if (b == &updateModuleButton) showUpdateModulesMenu();
|
|
else if (b == &setCopyModeButton) showCopyModeMenu();
|
|
else if (b == ©PathButton) showSetPathsMenu();
|
|
}
|
|
|
|
private:
|
|
enum
|
|
{
|
|
nameCol = 1,
|
|
versionCol,
|
|
updateCol,
|
|
copyCol,
|
|
pathCol
|
|
};
|
|
|
|
Project& project;
|
|
ValueTree modulesValueTree;
|
|
TableListBox table;
|
|
TextButton addWebModuleButton, updateModuleButton, setCopyModeButton, copyPathButton;
|
|
ScopedPointer<ModuleList> listFromWebsite;
|
|
|
|
void valueTreePropertyChanged (ValueTree&, const Identifier&) override { itemChanged(); }
|
|
void valueTreeChildAdded (ValueTree&, ValueTree&) override { itemChanged(); }
|
|
void valueTreeChildRemoved (ValueTree&, ValueTree&) override { itemChanged(); }
|
|
void valueTreeChildOrderChanged (ValueTree&) override { itemChanged(); }
|
|
void valueTreeParentChanged (ValueTree&) override { itemChanged(); }
|
|
|
|
void itemChanged()
|
|
{
|
|
table.updateContent();
|
|
resized();
|
|
repaint();
|
|
}
|
|
|
|
StringArray getUpdatableModules() const
|
|
{
|
|
StringArray result;
|
|
|
|
if (listFromWebsite != nullptr)
|
|
{
|
|
for (int i = 0; i < listFromWebsite->modules.size(); ++i)
|
|
{
|
|
const ModuleDescription* m = listFromWebsite->modules.getUnchecked(i);
|
|
const String v1 (m->getVersion());
|
|
const String v2 (project.getModules().getModuleInfo (m->getID()).getVersion());
|
|
|
|
if (v1 != v2 && v1.isNotEmpty() && v2.isNotEmpty())
|
|
result.add (m->getID());
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
StringArray getAddableModules() const
|
|
{
|
|
StringArray result;
|
|
|
|
if (listFromWebsite != nullptr)
|
|
{
|
|
for (int i = 0; i < listFromWebsite->modules.size(); ++i)
|
|
{
|
|
const ModuleDescription* m = listFromWebsite->modules.getUnchecked(i);
|
|
|
|
if (! project.getModules().isModuleEnabled (m->getID()))
|
|
result.add (m->getID());
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void showUpdateModulesMenu()
|
|
{
|
|
StringArray mods (getUpdatableModules());
|
|
|
|
PopupMenu m;
|
|
m.addItem (1000, "Update all modules");
|
|
m.addSeparator();
|
|
|
|
for (int i = 0; i < mods.size(); ++i)
|
|
m.addItem (1 + i, "Update " + mods[i]);
|
|
|
|
int res = m.showAt (&updateModuleButton);
|
|
|
|
if (res > 0 && listFromWebsite != nullptr)
|
|
{
|
|
if (res != 1000)
|
|
mods = StringArray (mods[res - 1]);
|
|
|
|
Array<ModuleDescription> modsToUpdate;
|
|
|
|
for (int i = 0; i < mods.size(); ++i)
|
|
{
|
|
if (const ModuleDescription* md = listFromWebsite->getModuleWithID (mods[i]))
|
|
{
|
|
ModuleDescription modToUpdate (*md);
|
|
modToUpdate.manifestFile = project.getModules().getModuleInfo (modToUpdate.getID()).manifestFile;
|
|
modsToUpdate.add (modToUpdate);
|
|
}
|
|
}
|
|
|
|
DownloadAndInstallThread::updateModulesFromWeb (project, modsToUpdate);
|
|
}
|
|
}
|
|
|
|
void showAddModuleMenu()
|
|
{
|
|
if (listFromWebsite == nullptr)
|
|
{
|
|
AlertWindow::showMessageBoxAsync (AlertWindow::WarningIcon,
|
|
"Couldn't contact the website!",
|
|
"Failed to get the latest module list from juce.com - "
|
|
"maybe network or server problems - try again soon!");
|
|
return;
|
|
}
|
|
|
|
StringArray mods (getAddableModules());
|
|
|
|
if (mods.size() == 0)
|
|
{
|
|
AlertWindow::showMessageBoxAsync (AlertWindow::WarningIcon,
|
|
"No modules to add!",
|
|
"Couldn't find any new modules that aren't already in your project!");
|
|
return;
|
|
}
|
|
|
|
PopupMenu m;
|
|
|
|
for (int i = 0; i < mods.size(); ++i)
|
|
m.addItem (i + 1, "Install " + mods[i]);
|
|
|
|
int res = m.showAt (&addWebModuleButton);
|
|
|
|
if (res > 0 && listFromWebsite != nullptr)
|
|
if (const ModuleDescription* md = listFromWebsite->getModuleWithID (mods[res - 1]))
|
|
DownloadAndInstallThread::addModuleFromWebsite (project, *md);
|
|
}
|
|
|
|
void showCopyModeMenu()
|
|
{
|
|
PopupMenu m;
|
|
m.addItem (1, "Set all modules to copy locally");
|
|
m.addItem (2, "Set all modules to not copy locally");
|
|
|
|
int res = m.showAt (&setCopyModeButton);
|
|
|
|
if (res != 0)
|
|
project.getModules().setLocalCopyModeForAllModules (res == 1);
|
|
}
|
|
|
|
void showSetPathsMenu()
|
|
{
|
|
EnabledModuleList& moduleList = project.getModules();
|
|
|
|
const String moduleToCopy (moduleList.getModuleID (table.getSelectedRow()));
|
|
|
|
if (moduleToCopy.isNotEmpty())
|
|
{
|
|
PopupMenu m;
|
|
m.addItem (1, "Copy the paths from the module '" + moduleToCopy + "' to all other modules");
|
|
|
|
int res = m.showAt (©PathButton);
|
|
|
|
if (res != 0)
|
|
{
|
|
for (Project::ExporterIterator exporter (project); exporter.next();)
|
|
{
|
|
for (int i = 0; i < moduleList.getNumModules(); ++i)
|
|
{
|
|
String modID = moduleList.getModuleID (i);
|
|
|
|
if (modID != moduleToCopy)
|
|
exporter->getPathForModuleValue (modID) = exporter->getPathForModuleValue (moduleToCopy).getValue();
|
|
}
|
|
}
|
|
}
|
|
|
|
table.repaint();
|
|
}
|
|
else
|
|
{
|
|
PopupMenu m;
|
|
m.addItem (1, "Copy the paths from the selected module to all other modules", false);
|
|
|
|
m.showAt (©PathButton);
|
|
}
|
|
}
|
|
|
|
struct WebsiteUpdateFetchThread : private Thread,
|
|
private AsyncUpdater
|
|
{
|
|
WebsiteUpdateFetchThread (ModulesPanel& p) : Thread ("Web Updater"), panel (p)
|
|
{
|
|
startThread (3);
|
|
}
|
|
|
|
~WebsiteUpdateFetchThread()
|
|
{
|
|
stopThread (15000);
|
|
}
|
|
|
|
void run() override
|
|
{
|
|
static Time lastDownloadTime;
|
|
static ModuleList lastList;
|
|
|
|
if (Time::getCurrentTime() < lastDownloadTime + RelativeTime::minutes (2.0))
|
|
{
|
|
list = lastList;
|
|
triggerAsyncUpdate();
|
|
}
|
|
else
|
|
{
|
|
if (list.loadFromWebsite() && ! threadShouldExit())
|
|
{
|
|
lastList = list;
|
|
lastDownloadTime = Time::getCurrentTime();
|
|
triggerAsyncUpdate();
|
|
}
|
|
}
|
|
}
|
|
|
|
void handleAsyncUpdate() override
|
|
{
|
|
panel.webUpdateFinished (list);
|
|
}
|
|
|
|
private:
|
|
ModuleList list;
|
|
ModulesPanel& panel;
|
|
};
|
|
|
|
ScopedPointer<WebsiteUpdateFetchThread> webUpdateThread;
|
|
|
|
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ModulesPanel)
|
|
};
|
|
|
|
//==============================================================================
|
|
class DownloadAndInstallThread : public ThreadWithProgressWindow
|
|
{
|
|
public:
|
|
DownloadAndInstallThread (const Array<ModuleDescription>& modulesToInstall)
|
|
: ThreadWithProgressWindow ("Installing New Modules", true, true),
|
|
result (Result::ok()),
|
|
modules (modulesToInstall)
|
|
{
|
|
}
|
|
|
|
static void updateModulesFromWeb (Project& project, const Array<ModuleDescription>& mods)
|
|
{
|
|
DownloadAndInstallThread d (mods);
|
|
|
|
if (d.runThread())
|
|
{
|
|
if (d.result.failed())
|
|
{
|
|
AlertWindow::showMessageBoxAsync (AlertWindow::WarningIcon,
|
|
"Module Install Failed",
|
|
d.result.getErrorMessage());
|
|
}
|
|
else
|
|
{
|
|
for (int i = 0; i < d.modules.size(); ++i)
|
|
project.getModules().addModule (d.modules.getReference(i).manifestFile,
|
|
project.getModules().areMostModulesCopiedLocally());
|
|
}
|
|
}
|
|
}
|
|
|
|
static void addModuleFromWebsite (Project& project, const ModuleDescription& module)
|
|
{
|
|
Array<ModuleDescription> mods;
|
|
mods.add (module);
|
|
|
|
static File lastLocation (EnabledModuleList::findDefaultModulesFolder (project));
|
|
|
|
FileChooser fc ("Select the parent folder for the new module...", lastLocation, String::empty, false);
|
|
|
|
if (fc.browseForDirectory())
|
|
{
|
|
lastLocation = fc.getResult();
|
|
|
|
if (lastLocation.getChildFile (ModuleDescription::getManifestFileName()).exists())
|
|
{
|
|
AlertWindow::showMessageBoxAsync (AlertWindow::WarningIcon,
|
|
"Adding Module",
|
|
"You chose a folder that appears to be a module.\n\n"
|
|
"You need to select the *parent* folder inside which the new modules will be created.");
|
|
return;
|
|
}
|
|
|
|
for (int i = 0; i < mods.size(); ++i)
|
|
mods.getReference(i).manifestFile = lastLocation.getChildFile (mods.getReference(i).getID())
|
|
.getChildFile (ModuleDescription::getManifestFileName());
|
|
|
|
updateModulesFromWeb (project, mods);
|
|
}
|
|
}
|
|
|
|
void run() override
|
|
{
|
|
for (int i = 0; i < modules.size(); ++i)
|
|
{
|
|
const ModuleDescription& m = modules.getReference(i);
|
|
|
|
setProgress (i / (double) modules.size());
|
|
|
|
MemoryBlock downloaded;
|
|
result = download (m, downloaded);
|
|
|
|
if (result.failed() || threadShouldExit())
|
|
break;
|
|
|
|
result = unzip (m, downloaded);
|
|
|
|
if (result.failed() || threadShouldExit())
|
|
break;
|
|
}
|
|
}
|
|
|
|
Result download (const ModuleDescription& m, MemoryBlock& dest)
|
|
{
|
|
setStatusMessage ("Downloading " + m.getID() + "...");
|
|
|
|
const ScopedPointer<InputStream> in (m.url.createInputStream (false, nullptr, nullptr, String::empty, 10000));
|
|
|
|
if (in != nullptr && in->readIntoMemoryBlock (dest))
|
|
return Result::ok();
|
|
|
|
return Result::fail ("Failed to download from: " + m.url.toString (false));
|
|
}
|
|
|
|
Result unzip (const ModuleDescription& m, const MemoryBlock& data)
|
|
{
|
|
setStatusMessage ("Installing " + m.getID() + "...");
|
|
|
|
MemoryInputStream input (data, false);
|
|
ZipFile zip (input);
|
|
|
|
if (zip.getNumEntries() == 0)
|
|
return Result::fail ("The downloaded file wasn't a valid module file!");
|
|
|
|
if (! m.getFolder().deleteRecursively())
|
|
return Result::fail ("Couldn't delete the existing folder:\n" + m.getFolder().getFullPathName());
|
|
|
|
return zip.uncompressTo (m.getFolder().getParentDirectory(), true);
|
|
}
|
|
|
|
Result result;
|
|
Array<ModuleDescription> modules;
|
|
|
|
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (DownloadAndInstallThread)
|
|
};
|