processing.app.Base Maven / Gradle / Ivy
Show all versions of pde Show documentation
/* -*- mode: java; c-basic-offset: 2; indent-tabs-mode: nil -*- */
/*
Part of the Processing project - http://processing.org
Copyright (c) 2012-16 The Processing Foundation
Copyright (c) 2004-12 Ben Fry and Casey Reas
Copyright (c) 2001-04 Massachusetts Institute of Technology
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
version 2, as published by the Free Software Foundation.
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package processing.app;
import java.awt.EventQueue;
import java.awt.FileDialog;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.*;
import java.lang.reflect.InvocationTargetException;
import java.text.SimpleDateFormat;
import java.util.*;
import javax.swing.JFileChooser;
import javax.swing.JMenu;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.JPopupMenu;
import javax.swing.tree.DefaultMutableTreeNode;
import processing.app.contrib.*;
import processing.app.tools.Tool;
import processing.app.ui.*;
import processing.core.*;
import processing.data.StringList;
/**
* The base class for the main processing application.
* Primary role of this class is for platform identification and
* general interaction with the system (launching URLs, loading
* files and images, etc) that comes from that.
*/
public class Base {
// Added accessors for 0218 because the UpdateCheck class was not properly
// updating the values, due to javac inlining the static final values.
static private final int REVISION = 253;
/** This might be replaced by main() if there's a lib/version.txt file. */
static private String VERSION_NAME = "0253"; //$NON-NLS-1$
/** Set true if this a proper release rather than a numbered revision. */
/** True if heavy debugging error/log messages are enabled */
static public boolean DEBUG = false;
// static public boolean DEBUG = true;
static private boolean commandLine;
// A single instance of the preferences window
PreferencesFrame preferencesFrame;
// A single instance of the library manager window
// ContributionManagerDialog contributionManagerFrame;
// Location for untitled items
static File untitledFolder;
/** List of currently active editors. */
protected List editors =
Collections.synchronizedList(new ArrayList());
protected Editor activeEditor;
/** A lone file menu to be used when all sketch windows are closed. */
static public JMenu defaultFileMenu;
/**
* Starts with the last mode used with the environment,
* or the default mode if not used.
*/
private Mode nextMode;
/** The built-in modes. coreModes[0] will be considered the 'default'. */
private Mode[] coreModes;
protected List modeContribs;
protected List exampleContribs;
private JMenu sketchbookMenu;
// private Recent recent;
// Used by handleOpen(), this saves the chooser to remember the directory.
// Doesn't appear to be necessary with the AWT native file dialog.
// https://github.com/processing/processing/pull/2366
private JFileChooser openChooser;
static protected File sketchbookFolder;
// protected File toolsFolder;
static public void main(final String[] args) {
EventQueue.invokeLater(new Runnable() {
public void run() {
try {
createAndShowGUI(args);
} catch (Throwable t) {
Messages.showTrace("It was not meant to be",
"A serious problem happened during startup. Please report:\n" +
"http://github.com/processing/processing/issues/new", t, true);
}
}
});
}
static private void createAndShowGUI(String[] args) {
try {
File versionFile = Platform.getContentFile("lib/version.txt");
if (versionFile.exists()) {
String version = PApplet.loadStrings(versionFile)[0];
if (!version.equals(VERSION_NAME)) {
VERSION_NAME = version;
// RELEASE = true;
}
}
} catch (Exception e) {
e.printStackTrace();
}
Platform.init();
// Use native popups so they don't look so crappy on OS X
JPopupMenu.setDefaultLightWeightPopupEnabled(false);
// Don't put anything above this line that might make GUI,
// because the platform has to be inited properly first.
// Make sure a full JDK is installed
//initRequirements();
// Load the languages
Language.init();
// run static initialization that grabs all the prefs
Preferences.init();
if (!SingleInstance.alreadyRunning(args)) {
// Set the look and feel before opening the window
try {
Platform.setLookAndFeel();
} catch (Exception e) {
Messages.loge("Could not set the Look & Feel", e); //$NON-NLS-1$
}
boolean sketchbookPrompt = false;
if (Preferences.getBoolean("welcome.show")) {
if (!Preferences.getBoolean("welcome.seen")) {
// Check if there's a 2.0 sketchbook present
String oldPath = Preferences.getOldSketchbookPath();
if (oldPath != null) {
String newPath = Preferences.getSketchbookPath();
// If newPath is null, this is the first run of any 3.x version
if (newPath == null) {
sketchbookPrompt = true;
} else if (oldPath.equals(newPath)) {
// If both exist and are identical, then the user has been using
// alpha releases of 3.x and needs to be warned about the larger
// changes in this release.
sketchbookPrompt = true;
}
}
}
}
// Get the sketchbook path, and make sure it's set properly
locateSketchbookFolder();
// Create a location for untitled sketches
try {
untitledFolder = Util.createTempFolder("untitled", "sketches", null);
untitledFolder.deleteOnExit();
} catch (IOException e) {
Messages.showError("Trouble without a name",
"Could not create a place to store untitled sketches.\n" +
"That's gonna prevent us from continuing.", e);
}
Messages.log("about to create base..."); //$NON-NLS-1$
try {
final Base base = new Base(args);
// Prevent more than one copy of the PDE from running.
SingleInstance.startServer(base);
// Needs to be shown after the first editor window opens, so that it
// shows up on top, and doesn't prevent an editor window from opening.
if (Preferences.getBoolean("welcome.show")) {
final boolean prompt = sketchbookPrompt;
EventQueue.invokeLater(new Runnable() {
public void run() {
try {
new Welcome(base, prompt);
} catch (IOException e) {
Messages.showTrace("Unwelcoming",
"Please report this error to\n" +
"https://github.com/processing/processing/issues", e, false);
}
}
});
}
} catch (Throwable t) {
// Catch-all to pick up badness during startup.
if (t.getCause() != null) {
// Usually this is the more important piece of information. We'll
// show this one so that it's not truncated in the error window.
t = t.getCause();
}
Messages.showTrace("We're off on the wrong foot",
"An error occurred during startup.", t, true);
}
Messages.log("done creating base..."); //$NON-NLS-1$
}
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
/**
* @return the current revision number, safe to be used for update checks
*/
static public int getRevision() {
return REVISION;
}
/**
* @return something like 2.2.1 or 3.0b4 (or 0213 if it's not a release)
*/
static public String getVersionName() {
return VERSION_NAME;
}
public static void setCommandLine() {
commandLine = true;
}
static public boolean isCommandLine() {
return commandLine;
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
public Base(String[] args) throws Exception {
ContributionManager.init(this);
buildCoreModes();
rebuildContribModes();
rebuildContribExamples();
// Needs to happen after the sketchbook folder has been located.
// Also relies on the modes to be loaded so it knows what can be
// marked as an example.
// recent = new Recent(this);
Recent.init(this);
String lastModeIdentifier = Preferences.get("mode.last"); //$NON-NLS-1$
if (lastModeIdentifier == null) {
nextMode = getDefaultMode();
Messages.log("Nothing set for last.sketch.mode, using default."); //$NON-NLS-1$
} else {
for (Mode m : getModeList()) {
if (m.getIdentifier().equals(lastModeIdentifier)) {
Messages.logf("Setting next mode to %s.", lastModeIdentifier); //$NON-NLS-1$
nextMode = m;
}
}
if (nextMode == null) {
nextMode = getDefaultMode();
Messages.logf("Could not find mode %s, using default.", lastModeIdentifier); //$NON-NLS-1$
}
}
//contributionManagerFrame = new ContributionManagerDialog();
// Make sure ThinkDifferent has library examples too
nextMode.rebuildLibraryList();
// Put this after loading the examples, so that building the default file
// menu works on Mac OS X (since it needs examplesFolder to be set).
Platform.initBase(this);
// toolsFolder = getContentFile("tools");
// // Check if there were previously opened sketches to be restored
// boolean opened = restoreSketches();
boolean opened = false;
// Check if any files were passed in on the command line
for (int i = 0; i < args.length; i++) {
String path = args[i];
// Fix a problem with systems that use a non-ASCII languages. Paths are
// being passed in with 8.3 syntax, which makes the sketch loader code
// unhappy, since the sketch folder naming doesn't match up correctly.
// http://dev.processing.org/bugs/show_bug.cgi?id=1089
if (Platform.isWindows()) {
try {
File file = new File(args[i]);
path = file.getCanonicalPath();
} catch (IOException e) {
e.printStackTrace();
}
}
if (handleOpen(path) != null) {
opened = true;
}
}
// Create a new empty window (will be replaced with any files to be opened)
if (!opened) {
// System.out.println("opening a new window");
handleNew();
// } else {
// System.out.println("something else was opened");
}
// check for updates
new UpdateCheck(this);
ContributionListing cl = ContributionListing.getInstance();
cl.downloadAvailableList(this, new ContribProgressMonitor() { });
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
void buildCoreModes() {
Mode javaMode =
ModeContribution.load(this, Platform.getContentFile("modes/java"),
getDefaultModeIdentifier()).getMode();
// PDE X calls getModeList() while it's loading, so coreModes must be set
coreModes = new Mode[] { javaMode };
/*
Mode pdexMode =
ModeContribution.load(this, getContentFile("modes/ExperimentalMode"), //$NON-NLS-1$
"processing.mode.experimental.ExperimentalMode").getMode(); //$NON-NLS-1$
// Safe to remove the old Java mode here?
//coreModes = new Mode[] { pdexMode };
coreModes = new Mode[] { pdexMode, javaMode };
*/
}
/**
* Instantiates and adds new contributed modes to the contribModes list.
* Checks for duplicates so the same mode isn't instantiates twice. Does not
* remove modes because modes can't be removed once they are instantiated.
*/
void rebuildContribModes() {
if (modeContribs == null) {
modeContribs = new ArrayList();
}
File modesFolder = getSketchbookModesFolder();
List contribModes = getModeContribs();
Map known = new HashMap();
for (ModeContribution contrib : contribModes) {
known.put(contrib.getFolder(), contrib);
}
File[] potential = ContributionType.MODE.listCandidates(modesFolder);
// If modesFolder does not exist or is inaccessible (folks might like to
// mess with folders then report it as a bug) 'potential' will be null.
if (potential != null) {
for (File folder : potential) {
if (!known.containsKey(folder)) {
try {
contribModes.add(new ModeContribution(this, folder, null));
} catch (NoSuchMethodError nsme) {
System.err.println(folder.getName() + " is not compatible with this version of Processing");
if (DEBUG) nsme.printStackTrace();
} catch (NoClassDefFoundError ncdfe) {
System.err.println(folder.getName() + " is not compatible with this version of Processing");
if (DEBUG) ncdfe.printStackTrace();
} catch (InvocationTargetException ite) {
System.err.println(folder.getName() + " could not be loaded and may not compatible with this version of Processing");
if (DEBUG) ite.printStackTrace();
} catch (IgnorableException ig) {
Messages.log(ig.getMessage());
if (DEBUG) ig.printStackTrace();
} catch (Throwable e) {
System.err.println("Could not load Mode from " + folder);
e.printStackTrace();
}
} else {
known.remove(folder); // remove this item as already been seen
}
}
}
// This allows you to build and test your Mode code from Eclipse.
// -Dusemode=com.foo.FrobMode:/path/to/FrobMode
final String useMode = System.getProperty("usemode");
if (useMode != null) {
final String[] modeInfo = useMode.split(":", 2);
final String modeClass = modeInfo[0];
final String modeResourcePath = modeInfo[1];
System.out.println("Attempting to load " + modeClass + " with resources at " + modeResourcePath);
ModeContribution mc = ModeContribution.load(this, new File(modeResourcePath), modeClass);
contribModes.add(mc);
known.remove(mc);
}
if (known.size() != 0) {
for (ModeContribution mc : known.values()) {
System.out.println("Extraneous Mode entry: " + mc.getName());
}
}
}
/**
* Instantiates and adds new contributed modes to the contribModes list.
* Checks for duplicates so the same mode isn't instantiates twice. Does not
* remove modes because modes can't be removed once they are instantiated.
*/
void rebuildContribExamples() {
if (exampleContribs == null) {
exampleContribs = new ArrayList();
}
ExamplesContribution.loadMissing(this);
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
/**
* Tools require an 'Editor' object when they're instantiated, but the
* activeEditor will be null when the first Editor that opens is creating
* its Tools menu. This will temporarily set the activeEditor to the one
* that's opening so that we don't go all NPE on startup. If there's already
* an active editor, then this does nothing.
*/
public void checkFirstEditor(Editor editor) {
if (activeEditor == null) {
activeEditor = editor;
}
}
/** Returns the front most, active editor window. */
public Editor getActiveEditor() {
return activeEditor;
}
/** Get the list of currently active editor windows. */
public List getEditors() {
return editors;
}
/**
* Called when a window is activated. Because of variations in native
* windowing systems, no guarantees about changes to the focused and active
* Windows can be made. Never assume that this Window is the focused or
* active Window until this Window actually receives a WINDOW_GAINED_FOCUS
* or WINDOW_ACTIVATED event.
*/
public void handleActivated(Editor whichEditor) {
activeEditor = whichEditor;
// set the current window to be the console that's getting output
EditorConsole.setEditor(activeEditor);
// make this the next mode to be loaded
nextMode = whichEditor.getMode();
Preferences.set("mode.last", nextMode.getIdentifier()); //$NON-NLS-1$
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
public void refreshContribs(ContributionType ct) {
if (ct == ContributionType.LIBRARY) {
for (Mode m : getModeList()) {
m.rebuildImportMenu();
}
} else if (ct == ContributionType.MODE) {
rebuildContribModes();
for (Editor editor : editors) {
editor.rebuildModePopup();
}
} else if (ct == ContributionType.TOOL) {
rebuildToolList();
for (Editor editor : editors) {
populateToolsMenu(editor.getToolMenu());
}
} else if (ct == ContributionType.EXAMPLES) {
rebuildContribExamples();
for (Mode m : getModeList()) {
m.rebuildExamplesFrame();
}
}
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
private int updatesAvailable = 0;
public void setUpdatesAvailable(int n) {
updatesAvailable = n;
synchronized (editors) {
for (Editor e : editors) {
e.setUpdatesAvailable(n);
}
}
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
List internalTools;
List coreTools;
List contribTools;
public List getCoreTools() {
return coreTools;
}
public List getToolContribs() {
return contribTools;
}
public void removeToolContrib(ToolContribution tc) {
contribTools.remove(tc);
}
public void rebuildToolList() {
// Only do this once because the list of internal tools will never change
if (internalTools == null) {
internalTools = new ArrayList();
initInternalTool("processing.app.tools.CreateFont");
initInternalTool("processing.app.tools.ColorSelector");
initInternalTool("processing.app.tools.Archiver");
if (Platform.isMacOS()) {
initInternalTool("processing.app.tools.InstallCommander");
}
}
// No need to reload these either
if (coreTools == null) {
coreTools = ToolContribution.loadAll(Base.getToolsFolder());
for (Tool tool : coreTools) {
tool.init(this);
}
}
// Rebuilt when new tools installed, etc
contribTools = ToolContribution.loadAll(Base.getSketchbookToolsFolder());
for (Tool tool : contribTools) {
try {
tool.init(this);
// With the exceptions, we can't call statusError because the window
// isn't completely set up yet. Also not gonna pop up a warning because
// people may still be running different versions of Processing.
} catch (VerifyError ve) {
System.err.println("\"" + tool.getMenuTitle() + "\" is not " +
"compatible with this version of Processing");
} catch (NoSuchMethodError nsme) {
System.err.println("\"" + tool.getMenuTitle() + "\" is not " +
"compatible with this version of Processing");
System.err.println("The " + nsme.getMessage() + " method no longer exists.");
Messages.loge("Incompatible Tool found during tool.init()", nsme);
} catch (NoClassDefFoundError ncdfe) {
System.err.println("\"" + tool.getMenuTitle() + "\" is not " +
"compatible with this version of Processing");
System.err.println("The " + ncdfe.getMessage() + " class is no longer available.");
Messages.loge("Incompatible Tool found during tool.init()", ncdfe);
} catch (AbstractMethodError ame) {
System.err.println("\"" + tool.getMenuTitle() + "\" is not " +
"compatible with this version of Processing");
// ame.printStackTrace();
} catch (Error err) {
System.err.println("An error occurred inside \"" + tool.getMenuTitle() + "\"");
err.printStackTrace();
} catch (Exception ex) {
System.err.println("An exception occurred inside \"" + tool.getMenuTitle() + "\"");
ex.printStackTrace();
}
}
}
protected void initInternalTool(String className) {
try {
Class> toolClass = Class.forName(className);
final Tool tool = (Tool) toolClass.newInstance();
tool.init(this);
internalTools.add(tool);
} catch (Exception e) {
e.printStackTrace();
}
}
/*
Iterator editorIter = base.getEditors().iterator();
while (editorIter.hasNext()) {
Editor editor = editorIter.next();
List contribTools = editor.getToolContribs();
for (ToolContribution toolContrib : contribTools) {
if (toolContrib.getName().equals(this.name)) {
try {
((URLClassLoader) toolContrib.loader).close();
editor.removeToolContrib(toolContrib);
break;
} catch (IOException e) {
e.printStackTrace();
}
}
}
*/
public void clearToolMenus() {
for (Editor ed : editors) {
ed.clearToolMenu();
}
}
public void populateToolsMenu(JMenu toolsMenu) {
// If this is the first run, need to build out the lists
if (internalTools == null) {
rebuildToolList();
}
// coreTools = ToolContribution.loadAll(Base.getToolsFolder());
// contribTools = ToolContribution.loadAll(Base.getSketchbookToolsFolder());
// Collections.sort(coreTools);
// Collections.sort(contribTools);
// Collections.sort(coreTools, new Comparator() {
// @Override
// public int compare(ToolContribution o1, ToolContribution o2) {
// return o1.getMenuTitle().compareTo(o2.getMenuTitle());
// }
// });
toolsMenu.removeAll();
for (Tool tool : internalTools) {
toolsMenu.add(createToolItem(tool));
}
toolsMenu.addSeparator();
if (coreTools.size() > 0) {
for (Tool tool : coreTools) {
toolsMenu.add(createToolItem(tool));
}
toolsMenu.addSeparator();
}
if (contribTools.size() > 0) {
for (Tool tool : contribTools) {
toolsMenu.add(createToolItem(tool));
}
toolsMenu.addSeparator();
}
JMenuItem item = new JMenuItem(Language.text("menu.tools.add_tool"));
item.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
ContributionManager.openTools();
}
});
toolsMenu.add(item);
}
/*
static public void addTools(JMenu menu, List tools) {
Map toolItems = new HashMap();
for (final Tool tool : tools) {
// If init() fails, the item won't be added to the menu
addToolItem(tool, toolItems);
}
List toolList = new ArrayList(toolItems.keySet());
if (toolList.size() > 0) {
if (menu.getItemCount() != 0) {
menu.addSeparator();
}
Collections.sort(toolList);
for (String title : toolList) {
menu.add(toolItems.get(title));
}
}
}
*/
JMenuItem createToolItem(final Tool tool) { //, Map toolItems) {
String title = tool.getMenuTitle();
final JMenuItem item = new JMenuItem(title);
item.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
try {
tool.run();
} catch (NoSuchMethodError nsme) {
activeEditor.statusError("\"" + tool.getMenuTitle() + "\" is not" +
"compatible with this version of Processing");
//nsme.printStackTrace();
Messages.loge("Incompatible tool found during tool.run()", nsme);
item.setEnabled(false);
} catch (Exception ex) {
activeEditor.statusError("An error occurred inside \"" + tool.getMenuTitle() + "\"");
ex.printStackTrace();
item.setEnabled(false);
}
}
});
//toolItems.put(title, item);
return item;
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
public List getModeContribs() {
return modeContribs;
}
public List getModeList() {
List allModes = new ArrayList();
allModes.addAll(Arrays.asList(coreModes));
if (modeContribs != null) {
for (ModeContribution contrib : modeContribs) {
allModes.add(contrib.getMode());
}
}
return allModes;
}
public List getExampleContribs() {
return exampleContribs;
}
private List getInstalledContribs() {
List contributions = new ArrayList();
List modeContribs = getModeContribs();
contributions.addAll(modeContribs);
for (ModeContribution modeContrib : modeContribs) {
Mode mode = modeContrib.getMode();
contributions.addAll(new ArrayList(mode.contribLibraries));
}
// TODO this duplicates code in Editor, but it's not editor-specific
// List toolContribs =
// ToolContribution.loadAll(Base.getSketchbookToolsFolder());
// contributions.addAll(toolContribs);
contributions.addAll(ToolContribution.loadAll(getSketchbookToolsFolder()));
contributions.addAll(getExampleContribs());
return contributions;
}
public byte[] getInstalledContribsInfo() {
List contribs = getInstalledContribs();
StringList entries = new StringList();
for (Contribution c : contribs) {
String entry = c.getTypeName() + "=" +
PApplet.urlEncode(String.format("name=%s\nurl=%s\nrevision=%d\nversion=%s",
c.getName(), c.getUrl(),
c.getVersion(), c.getPrettyVersion()));
entries.append(entry);
}
String joined =
"id=" + Preferences.get("update.id") + "&" + entries.join("&");
// StringBuilder sb = new StringBuilder();
// try {
// // Truly ridiculous attempt to shove everything into a GET request.
// // More likely to be seen as part of a grand plot.
// ByteArrayOutputStream baos = new ByteArrayOutputStream();
// GZIPOutputStream output = new GZIPOutputStream(baos);
// PApplet.saveStream(output, new ByteArrayInputStream(joined.getBytes()));
// output.close();
// byte[] b = baos.toByteArray();
// for (int i = 0; i < b.length; i++) {
// sb.append(PApplet.hex(b[i], 2));
// }
// } catch (IOException e) {
// e.printStackTrace();
// }
// return sb.toString();
return joined.getBytes();
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
/**
* Create or modify a sketch.proprties file to specify the given Mode.
*/
private void saveModeSettings(final File sketchProps, final Mode mode) {
try {
final Settings settings = new Settings(sketchProps);
settings.set("mode", mode.getTitle());
settings.set("mode.id", mode.getIdentifier());
settings.save();
} catch (IOException e) {
System.err.println("While creating " + sketchProps + ": " + e.getMessage());
}
}
String getDefaultModeIdentifier() {
return "processing.mode.java.JavaMode";
}
public Mode getDefaultMode() {
return coreModes[0];
}
/** Used by ThinkDifferent so that it can have a Sketchbook menu. */
public Mode getNextMode() {
return nextMode;
}
/**
* The call has already checked to make sure this sketch is not modified,
* now change the mode.
* @return true if mode is changed.
*/
public boolean changeMode(Mode mode) {
Mode oldMode = activeEditor.getMode();
if (oldMode != mode) {
Sketch sketch = activeEditor.getSketch();
nextMode = mode;
if (sketch.isUntitled()) {
// The current sketch is empty, just close and start fresh.
// (Otherwise the editor would lose its 'untitled' status.)
handleClose(activeEditor, true);
handleNew();
} else {
// If the current editor contains file extensions that the new mode can handle, then
// write a sketch.properties file with the new mode specified, and reopen.
boolean newModeCanHandleCurrentSource = true;
for (final SketchCode code : sketch.getCode()) {
if (!mode.validExtension(code.getExtension())) {
newModeCanHandleCurrentSource = false;
break;
}
}
if (!newModeCanHandleCurrentSource) {
return false;
} else {
final File props = new File(sketch.getCodeFolder(), "sketch.properties");
saveModeSettings(props, nextMode);
handleClose(activeEditor, true);
Editor editor = handleOpen(sketch.getMainFilePath());
if (editor == null) {
// the Mode change failed (probably code that's out of date)
// re-open the sketch using the mode we were in before
saveModeSettings(props, oldMode);
handleOpen(sketch.getMainFilePath());
return false;
}
}
}
}
return true;
}
private static class ModeInfo {
public final String title;
public final String id;
public ModeInfo(String id, String title) {
this.id = id;
this.title = title;
}
}
private static ModeInfo modeInfoFor(final File sketch) {
final File sketchFolder = sketch.getParentFile();
final File sketchProps = new File(sketchFolder, "sketch.properties");
if (!sketchProps.exists()) {
return null;
}
try {
final Settings settings = new Settings(sketchProps);
final String title = settings.get("mode");
final String id = settings.get("mode.id");
if (title == null || id == null) {
return null;
}
return new ModeInfo(id, title);
} catch (IOException e) {
System.err.println("While trying to read " + sketchProps + ": "
+ e.getMessage());
}
return null;
}
private Mode promptForMode(final File sketch, final ModeInfo preferredMode) {
final String extension =
sketch.getName().substring(sketch.getName().lastIndexOf('.') + 1);
final List possibleModes = new ArrayList();
for (final Mode mode : getModeList()) {
if (mode.canEdit(sketch)) {
possibleModes.add(mode);
}
}
if (possibleModes.size() == 1 &&
possibleModes.get(0).getIdentifier().equals(getDefaultModeIdentifier())) {
// If default mode can open it, then do so without prompting.
return possibleModes.get(0);
}
if (possibleModes.size() == 0) {
if (preferredMode == null) {
final String msg =
"I don't know how to open a sketch with the \"" + extension + "\"\n" +
"file extension. You'll have to install a different\n" +
"Mode for that.";
Messages.showWarning("Modeless Dialog", msg);
} else {
Messages.showWarning("Modeless Dialog",
"Install " + preferredMode.title + " Mode " +
"to open this sketch.");
}
return null;
}
final Mode[] modes = possibleModes.toArray(new Mode[possibleModes.size()]);
final String message = preferredMode == null ?
(nextMode.getTitle() + " Mode can't open ." + extension + " files, " +
"but you have one or more modes\ninstalled that can. " +
"Would you like to try one?") :
("That's a " + preferredMode.title + " Mode sketch, " +
"but you don't have " + preferredMode.title + " installed.\n" +
"Would you like to try a different mode for opening a " +
"." + extension + " sketch?");
return (Mode) JOptionPane.showInputDialog(null, message, "Choose Wisely",
JOptionPane.QUESTION_MESSAGE,
null, modes, modes[0]);
}
private Mode selectMode(final File sketch) {
final ModeInfo modeInfo = modeInfoFor(sketch);
final Mode specifiedMode = modeInfo == null ? null : findMode(modeInfo.id);
if (specifiedMode != null) {
return specifiedMode;
}
return promptForMode(sketch, modeInfo);
}
protected Mode findMode(String id) {
for (Mode mode : getModeList()) {
if (mode.getIdentifier().equals(id)) {
return mode;
}
}
return null;
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
boolean breakTime = false;
String[] months = {
"jan", "feb", "mar", "apr", "may", "jun",
"jul", "aug", "sep", "oct", "nov", "dec"
};
/**
* Create a new untitled document in a new sketch window.
*/
public void handleNew() {
try {
File newbieDir = null;
String newbieName = null;
// In 0126, untitled sketches will begin in the temp folder,
// and then moved to a new location because Save will default to Save As.
// File sketchbookDir = getSketchbookFolder();
File newbieParentDir = untitledFolder;
String prefix = Preferences.get("editor.untitled.prefix");
// Use a generic name like sketch_031008a, the date plus a char
int index = 0;
String format = Preferences.get("editor.untitled.suffix");
String suffix = null;
if (format == null) {
Calendar cal = Calendar.getInstance();
int day = cal.get(Calendar.DAY_OF_MONTH); // 1..31
int month = cal.get(Calendar.MONTH); // 0..11
suffix = months[month] + PApplet.nf(day, 2);
} else {
//SimpleDateFormat formatter = new SimpleDateFormat("yyMMdd");
//SimpleDateFormat formatter = new SimpleDateFormat("MMMdd");
//String purty = formatter.format(new Date()).toLowerCase();
SimpleDateFormat formatter = new SimpleDateFormat(format);
suffix = formatter.format(new Date());
}
do {
if (index == 26) {
// In 0159, avoid running past z by sending people outdoors.
if (!breakTime) {
Messages.showWarning("Time for a Break",
"You've reached the limit for auto naming of new sketches\n" +
"for the day. How about going for a walk instead?", null);
breakTime = true;
} else {
Messages.showWarning("Sunshine",
"No really, time for some fresh air for you.", null);
}
return;
}
newbieName = prefix + suffix + ((char) ('a' + index));
// Also sanitize the name since it might do strange things on
// non-English systems that don't use this sort of date format.
// http://code.google.com/p/processing/issues/detail?id=283
newbieName = Sketch.sanitizeName(newbieName);
newbieDir = new File(newbieParentDir, newbieName);
index++;
// Make sure it's not in the temp folder *and* it's not in the sketchbook
} while (newbieDir.exists() || new File(sketchbookFolder, newbieName).exists());
// Make the directory for the new sketch
newbieDir.mkdirs();
// Add any template files from the Mode itself
File newbieFile = nextMode.addTemplateFiles(newbieDir, newbieName);
/*
// Make an empty pde file
File newbieFile =
new File(newbieDir, newbieName + "." + nextMode.getDefaultExtension()); //$NON-NLS-1$
if (!newbieFile.createNewFile()) {
throw new IOException(newbieFile + " already exists.");
}
*/
// Create sketch properties file if it's not the default mode.
if (!nextMode.equals(getDefaultMode())) {
saveModeSettings(new File(newbieDir, "sketch.properties"), nextMode);
}
String path = newbieFile.getAbsolutePath();
/*Editor editor =*/ handleOpen(path, true);
} catch (IOException e) {
Messages.showWarning("That's new to me",
"A strange and unexplainable error occurred\n" +
"while trying to create a new sketch.", e);
}
}
/**
* Prompt for a sketch to open, and open it in a new window.
*/
public void handleOpenPrompt() {
final StringList extensions = new StringList();
for (Mode mode : getModeList()) {
extensions.append(mode.getDefaultExtension());
}
final String prompt = Language.text("open");
// don't use native dialogs on Linux (or anyone else w/ override)
if (Preferences.getBoolean("chooser.files.native")) { //$NON-NLS-1$
// use the front-most window frame for placing file dialog
FileDialog openDialog =
new FileDialog(activeEditor, prompt, FileDialog.LOAD);
// Only show .pde files as eligible bachelors
openDialog.setFilenameFilter(new FilenameFilter() {
public boolean accept(File dir, String name) {
// confirmed to be working properly [fry 110128]
for (String ext : extensions) {
if (name.toLowerCase().endsWith("." + ext)) { //$NON-NLS-1$
return true;
}
}
return false;
}
});
openDialog.setVisible(true);
String directory = openDialog.getDirectory();
String filename = openDialog.getFile();
if (filename != null) {
File inputFile = new File(directory, filename);
handleOpen(inputFile.getAbsolutePath());
}
} else {
if (openChooser == null) {
openChooser = new JFileChooser();
}
openChooser.setDialogTitle(prompt);
openChooser.setFileFilter(new javax.swing.filechooser.FileFilter() {
public boolean accept(File file) {
// JFileChooser requires you to explicitly say yes to directories
// as well (unlike the AWT chooser). Useful, but... different.
// http://code.google.com/p/processing/issues/detail?id=1151
if (file.isDirectory()) {
return true;
}
for (String ext : extensions) {
if (file.getName().toLowerCase().endsWith("." + ext)) { //$NON-NLS-1$
return true;
}
}
return false;
}
public String getDescription() {
return "Processing Sketch";
}
});
if (openChooser.showOpenDialog(activeEditor) == JFileChooser.APPROVE_OPTION) {
handleOpen(openChooser.getSelectedFile().getAbsolutePath());
}
}
}
/**
* Open a sketch from the path specified. Do not use for untitled sketches.
*/
public Editor handleOpen(String path) {
return handleOpen(path, false);
}
/**
* Open a sketch in a new window.
* @param path Path to the pde file for the sketch in question
* @return the Editor object, so that properties (like 'untitled')
* can be set by the caller
*/
public Editor handleOpen(String path, boolean untitled) {
return handleOpen(path, untitled, new EditorState(editors));
}
protected Editor handleOpen(String path, boolean untitled,
EditorState state) {
try {
// System.err.println("entering handleOpen " + path);
final File file = new File(path);
if (!file.exists()) {
return null;
}
// Cycle through open windows to make sure that it's not already open.
for (Editor editor : editors) {
// User may have double-clicked any PDE in the sketch folder,
// so we have to check each open tab (not just the main one).
// https://github.com/processing/processing/issues/2506
for (SketchCode tab : editor.getSketch().getCode()) {
if (tab.getFile().equals(file)) {
editor.toFront();
// move back to the top of the recent list
Recent.append(editor);
return editor;
}
}
}
if (!Sketch.isSanitaryName(file.getName())) {
Messages.showWarning("You're tricky, but not tricky enough",
file.getName() + " is not a valid name for a sketch.\n" +
"Better to stick to ASCII, no spaces, and make sure\n" +
"it doesn't start with a number.", null);
return null;
}
if (!nextMode.canEdit(file)) {
final Mode mode = selectMode(file);
if (mode == null) {
return null;
}
nextMode = mode;
}
try {
Editor editor = nextMode.createEditor(this, path, state);
editor.setUpdatesAvailable(updatesAvailable);
// opened successfully, let's go to work
editor.getSketch().setUntitled(untitled);
editors.add(editor);
Recent.append(editor);
// now that we're ready, show the window
// (don't do earlier, cuz we might move it based on a window being closed)
editor.setVisible(true);
return editor;
} catch (EditorException ee) {
if (ee.getMessage() != null) { // null if the user canceled
Messages.showWarning("Error opening sketch", ee.getMessage(), ee);
}
} catch (NoSuchMethodError nsme) {
Messages.showWarning("Mode out of date",
nextMode.getTitle() + " is not compatible with this version of Processing.\n" +
"Try updating the Mode or contact its author for a new version.", nsme);
} catch (Throwable t) {
if (nextMode.equals(getDefaultMode())) {
Messages.showTrace("Serious Problem",
"An unexpected, unknown, and unrecoverable error occurred\n" +
"while opening a new editor window. Please report this.", t, true);
} else {
Messages.showTrace("Mode Problems",
"A nasty error occurred while trying to use " + nextMode.getTitle() + ".\n" +
"It may not be compatible with this version of Processing.\n" +
"Try updating the Mode or contact its author for a new version.", t, false);
}
}
if (editors.isEmpty()) {
Mode defaultMode = getDefaultMode();
if (nextMode == defaultMode) {
// unreachable? hopefully?
Messages.showError("Editor Problems",
"An error occurred while trying to change modes.\n" +
"We'll have to quit for now because it's an\n" +
"unfortunate bit of indigestion with the default Mode.",
null);
} else {
// Don't leave the user hanging or the PDE locked up
// https://github.com/processing/processing/issues/4467
if (untitled) {
nextMode = defaultMode;
handleNew();
return null; // ignored by any caller
} else {
// This null response will be kicked back to changeMode(),
// signaling it to re-open the sketch in the default Mode.
return null;
}
}
}
/*
if (editors.isEmpty()) {
// if the bad mode is the default mode, don't go into an infinite loop
// trying to recreate a window with the default mode.
Mode defaultMode = getDefaultMode();
if (nextMode == defaultMode) {
Base.showError("Editor Problems",
"An error occurred while trying to change modes.\n" +
"We'll have to quit for now because it's an\n" +
"unfortunate bit of indigestion with the default Mode.",
null);
} else {
editor = defaultMode.createEditor(this, path, state);
}
}
*/
} catch (Throwable t) {
Messages.showTrace("Terrible News",
"A serious error occurred while " +
"trying to create a new editor window.", t,
nextMode == getDefaultMode()); // quit if default
nextMode = getDefaultMode();
}
return null;
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
/**
* Close a sketch as specified by its editor window.
* @param editor Editor object of the sketch to be closed.
* @param modeSwitch Whether this close is being done in the context of a
* mode switch.
* @return true if succeeded in closing, false if canceled.
*/
public boolean handleClose(Editor editor, boolean modeSwitch) {
// Check if modified
// boolean immediate = editors.size() == 1;
if (!editor.checkModified()) {
return false;
}
// Close the running window, avoid window boogers with multiple sketches
editor.internalCloseRunner();
// System.out.println("editors size is " + editors.size());
if (editors.size() == 1) {
// For 0158, when closing the last window /and/ it was already an
// untitled sketch, just give up and let the user quit.
// if (Preferences.getBoolean("sketchbook.closing_last_window_quits") ||
// (editor.untitled && !editor.getSketch().isModified())) {
if (Platform.isMacOS()) {
// If the central menubar isn't supported on this OS X JVM,
// we have to do the old behavior. Yuck!
if (defaultFileMenu == null) {
Object[] options = { Language.text("prompt.ok"), Language.text("prompt.cancel") };
String prompt =
" " +
" " +
"Are you sure you want to Quit?" +
"Closing the last open sketch will quit Processing.";
int result = JOptionPane.showOptionDialog(editor,
prompt,
"Quit",
JOptionPane.YES_NO_OPTION,
JOptionPane.QUESTION_MESSAGE,
null,
options,
options[0]);
if (result == JOptionPane.NO_OPTION ||
result == JOptionPane.CLOSED_OPTION) {
return false;
}
}
}
Preferences.unset("server.port"); //$NON-NLS-1$
Preferences.unset("server.key"); //$NON-NLS-1$
// // This will store the sketch count as zero
// editors.remove(editor);
// System.out.println("editors size now " + editors.size());
// storeSketches();
// Save out the current prefs state
Preferences.save();
if (defaultFileMenu == null) {
if (modeSwitch) {
// need to close this editor, ever so temporarily
editor.setVisible(false);
editor.dispose();
activeEditor = null;
editors.remove(editor);
} else {
// Since this wasn't an actual Quit event, call System.exit()
System.exit(0);
}
} else { // on OS X, update the default file menu
editor.setVisible(false);
editor.dispose();
defaultFileMenu.insert(Recent.getMenu(), 2);
activeEditor = null;
editors.remove(editor);
}
} else {
// More than one editor window open,
// proceed with closing the current window.
editor.setVisible(false);
editor.dispose();
editors.remove(editor);
}
return true;
}
/**
* Handler for File → Quit.
* @return false if canceled, true otherwise.
*/
public boolean handleQuit() {
// If quit is canceled, this will be replaced anyway
// by a later handleQuit() that is not canceled.
// storeSketches();
if (handleQuitEach()) {
// make sure running sketches close before quitting
for (Editor editor : editors) {
editor.internalCloseRunner();
}
// Save out the current prefs state
Preferences.save();
if (!Platform.isMacOS()) {
// If this was fired from the menu or an AppleEvent (the Finder),
// then Mac OS X will send the terminate signal itself.
System.exit(0);
}
return true;
}
return false;
}
/**
* Attempt to close each open sketch in preparation for quitting.
* @return false if canceled along the way
*/
protected boolean handleQuitEach() {
// int index = 0;
for (Editor editor : editors) {
// if (editor.checkModified()) {
// // Update to the new/final sketch path for this fella
// storeSketchPath(editor, index);
// index++;
//
// } else {
// return false;
// }
if (!editor.checkModified()) {
return false;
}
}
return true;
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
/**
* Asynchronous version of menu rebuild to be used on save and rename
* to prevent the interface from locking up until the menus are done.
*/
protected void rebuildSketchbookMenusAsync() {
EventQueue.invokeLater(new Runnable() {
public void run() {
rebuildSketchbookMenus();
}
});
}
public void thinkDifferentExamples() {
nextMode.showExamplesFrame();
}
/**
* Synchronous version of rebuild, used when the sketchbook folder has
* changed, so that the libraries are properly re-scanned before those menus
* (and the examples window) are rebuilt.
*/
protected void rebuildSketchbookMenus() {
for (Mode mode : getModeList()) {
mode.rebuildImportMenu(); // calls rebuildLibraryList
mode.rebuildToolbarMenu();
mode.rebuildExamplesFrame();
mode.rebuildSketchbookFrame();
}
}
protected void rebuildSketchbookMenu() {
sketchbookMenu.removeAll();
populateSketchbookMenu(sketchbookMenu);
}
public void populateSketchbookMenu(JMenu menu) {
boolean found = false;
try {
found = addSketches(menu, sketchbookFolder, false);
} catch (IOException e) {
Messages.showWarning("Sketchbook Menu Error",
"An error occurred while trying to list the sketchbook.", e);
}
if (!found) {
JMenuItem empty = new JMenuItem(Language.text("menu.file.sketchbook.empty"));
empty.setEnabled(false);
menu.add(empty);
}
}
/*
public JMenu getRecentMenu() {
return recent.getMenu();
}
public JMenu getToolbarRecentMenu() {
return recent.getToolbarMenu();
}
public void handleRecent(Editor editor) {
recent.handle(editor);
}
public void handleRecentRename(Editor editor, String oldPath) {
recent.handleRename(editor, oldPath);
}
// Called before a sketch is renamed so that its old name is
// no longer in the menu.
public void removeRecent(Editor editor) {
recent.remove(editor);
}
*/
/**
* Scan a folder recursively, and add any sketches found to the menu
* specified. Set the openReplaces parameter to true when opening the sketch
* should replace the sketch in the current window, or false when the
* sketch should open in a new window.
*/
protected boolean addSketches(JMenu menu, File folder,
final boolean replaceExisting) throws IOException {
// skip .DS_Store files, etc (this shouldn't actually be necessary)
if (!folder.isDirectory()) {
return false;
}
if (folder.getName().equals("libraries")) {
return false; // let's not go there
}
String[] list = folder.list();
// If a bad folder or unreadable or whatever, this will come back null
if (list == null) {
return false;
}
// Alphabetize the list, since it's not always alpha order
Arrays.sort(list, String.CASE_INSENSITIVE_ORDER);
ActionListener listener = new ActionListener() {
public void actionPerformed(ActionEvent e) {
String path = e.getActionCommand();
if (new File(path).exists()) {
boolean replace = replaceExisting;
if ((e.getModifiers() & ActionEvent.SHIFT_MASK) != 0) {
replace = !replace;
}
// if (replace) {
// handleOpenReplace(path);
// } else {
handleOpen(path);
// }
} else {
Messages.showWarning("Sketch Disappeared",
"The selected sketch no longer exists.\n" +
"You may need to restart Processing to update\n" +
"the sketchbook menu.", null);
}
}
};
// offers no speed improvement
//menu.addActionListener(listener);
boolean found = false;
// for (int i = 0; i < list.length; i++) {
// if ((list[i].charAt(0) == '.') ||
// list[i].equals("CVS")) continue;
for (String name : list) {
if (name.charAt(0) == '.') {
continue;
}
File subfolder = new File(folder, name);
if (subfolder.isDirectory()) {
File entry = checkSketchFolder(subfolder, name);
if (entry != null) {
JMenuItem item = new JMenuItem(name);
item.addActionListener(listener);
item.setActionCommand(entry.getAbsolutePath());
menu.add(item);
found = true;
} else {
// not a sketch folder, but maybe a subfolder containing sketches
JMenu submenu = new JMenu(name);
// needs to be separate var otherwise would set ifound to false
boolean anything = addSketches(submenu, subfolder, replaceExisting);
if (anything && !name.equals("old")) { //Don't add old contributions
menu.add(submenu);
found = true;
}
}
}
}
return found;
}
public boolean addSketches(DefaultMutableTreeNode node, File folder,
boolean examples) throws IOException {
// skip .DS_Store files, etc (this shouldn't actually be necessary)
if (!folder.isDirectory()) {
return false;
}
final String folderName = folder.getName();
// Don't look inside the 'libraries' folders in the sketchbook
if (folderName.equals("libraries")) {
return false;
}
// When building the sketchbook, don't show the contributed 'examples'
// like it's a subfolder. But when loading examples, allow the folder
// to be named 'examples'.
if (!examples && folderName.equals("examples")) {
return false;
}
// // Conversely, when looking for examples, ignore the other folders
// // (to avoid going through hoops with the tree node setup).
// if (examples && !folderName.equals("examples")) {
// return false;
// }
// // Doesn't quite work because the parent will be 'examples', and we want
// // to walk inside that, but the folder itself will have a different name
String[] fileList = folder.list();
// If a bad folder or unreadable or whatever, this will come back null
if (fileList == null) {
return false;
}
// Alphabetize the list, since it's not always alpha order
Arrays.sort(fileList, String.CASE_INSENSITIVE_ORDER);
boolean found = false;
for (String name : fileList) {
if (name.charAt(0) == '.') { // Skip hidden files
continue;
}
File subfolder = new File(folder, name);
if (subfolder.isDirectory()) {
File entry = checkSketchFolder(subfolder, name);
if (entry != null) {
DefaultMutableTreeNode item =
new DefaultMutableTreeNode(new SketchReference(name, entry));
node.add(item);
found = true;
} else {
// not a sketch folder, but maybe a subfolder containing sketches
DefaultMutableTreeNode subnode = new DefaultMutableTreeNode(name);
// needs to be separate var otherwise would set ifound to false
boolean anything = addSketches(subnode, subfolder, examples);
if (anything) {
node.add(subnode);
found = true;
}
}
}
}
return found;
}
/**
* Check through the various modes and see if this is a legit sketch.
* Because the default mode will be the first in the list, this will always
* prefer that one over the others.
*/
File checkSketchFolder(File subfolder, String item) {
for (Mode mode : getModeList()) {
File entry = new File(subfolder, item + "." + mode.getDefaultExtension()); //$NON-NLS-1$
// if a .pde file of the same prefix as the folder exists..
if (entry.exists()) {
return entry;
}
}
return null;
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
/**
* Show the Preferences window.
*/
public void handlePrefs() {
if (preferencesFrame == null) {
preferencesFrame = new PreferencesFrame(this);
}
preferencesFrame.showFrame();
}
// . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
/**
* Return a File from inside the Processing 'lib' folder.
*/
static public File getLibFile(String filename) throws IOException {
return new File(Platform.getContentFile("lib"), filename);
}
/**
* Return an InputStream for a file inside the Processing lib folder.
*/
static public InputStream getLibStream(String filename) throws IOException {
return new FileInputStream(getLibFile(filename));
}
/**
* Get the directory that can store settings. (Library on OS X, App Data or
* something similar on Windows, a dot folder on Linux.) Removed this as a
* preference for 3.0a3 because we need this to be stable.
*/
static public File getSettingsFolder() {
File settingsFolder = null;
try {
settingsFolder = Platform.getSettingsFolder();
// create the folder if it doesn't exist already
if (!settingsFolder.exists()) {
if (!settingsFolder.mkdirs()) {
Messages.showError("Settings issues",
"Processing cannot run because it could not\n" +
"create a folder to store your settings.\n" +
settingsFolder.getAbsolutePath(), null);
}
}
} catch (Exception e) {
Messages.showError("Problem getting the settings folder",
"Error getting the Processing the settings folder.", e);
}
return settingsFolder;
}
/**
* Convenience method to get a File object for the specified filename inside
* the settings folder. Used to get preferences and recent sketch files.
* @param filename A file inside the settings folder.
* @return filename wrapped as a File object inside the settings folder
*/
static public File getSettingsFile(String filename) {
return new File(getSettingsFolder(), filename);
}
static public File getToolsFolder() {
return Platform.getContentFile("tools");
}
static public void locateSketchbookFolder() {
// If a value is at least set, first check to see if the folder exists.
// If it doesn't, warn the user that the sketchbook folder is being reset.
String sketchbookPath = Preferences.getSketchbookPath();
if (sketchbookPath != null) {
sketchbookFolder = new File(sketchbookPath);
if (!sketchbookFolder.exists()) {
Messages.showWarning("Sketchbook folder disappeared",
"The sketchbook folder no longer exists.\n" +
"Processing will switch to the default sketchbook\n" +
"location, and create a new sketchbook folder if\n" +
"necessary. Processing will then stop talking\n" +
"about itself in the third person.", null);
sketchbookFolder = null;
}
}
// If no path is set, get the default sketchbook folder for this platform
if (sketchbookFolder == null) {
sketchbookFolder = getDefaultSketchbookFolder();
Preferences.setSketchbookPath(sketchbookFolder.getAbsolutePath());
if (!sketchbookFolder.exists()) {
sketchbookFolder.mkdirs();
}
}
makeSketchbookSubfolders();
}
public void setSketchbookFolder(File folder) {
sketchbookFolder = folder;
Preferences.setSketchbookPath(folder.getAbsolutePath());
rebuildSketchbookMenus();
makeSketchbookSubfolders();
}
/**
* Create the libraries, modes, tools, examples folders in the sketchbook.
*/
static protected void makeSketchbookSubfolders() {
getSketchbookLibrariesFolder().mkdirs();
getSketchbookToolsFolder().mkdirs();
getSketchbookModesFolder().mkdirs();
getSketchbookExamplesFolder().mkdirs();
getSketchbookTemplatesFolder().mkdirs();
}
static public File getSketchbookFolder() {
return sketchbookFolder;
}
static public File getSketchbookLibrariesFolder() {
return new File(sketchbookFolder, "libraries");
}
static public File getSketchbookToolsFolder() {
return new File(sketchbookFolder, "tools");
}
static public File getSketchbookModesFolder() {
return new File(sketchbookFolder, "modes");
}
static public File getSketchbookExamplesFolder() {
return new File(sketchbookFolder, "examples");
}
static public File getSketchbookTemplatesFolder() {
return new File(sketchbookFolder, "templates");
}
static protected File getDefaultSketchbookFolder() {
File sketchbookFolder = null;
try {
sketchbookFolder = Platform.getDefaultSketchbookFolder();
} catch (Exception e) { }
if (sketchbookFolder == null) {
Messages.showError("No sketchbook",
"Problem while trying to get the sketchbook", null);
}
// create the folder if it doesn't exist already
boolean result = true;
if (!sketchbookFolder.exists()) {
result = sketchbookFolder.mkdirs();
}
if (!result) {
Messages.showError("You forgot your sketchbook",
"Processing cannot run because it could not\n" +
"create a folder to store your sketchbook.", null);
}
return sketchbookFolder;
}
}