From b60d6b65eccc95df89f513db257fb0045ac0a9c6 Mon Sep 17 00:00:00 2001 From: Adam Sotona <10807609+asotona@users.noreply.github.com> Date: Thu, 10 Oct 2024 15:30:30 +0200 Subject: [PATCH] Single java source run action provider improvements --- extide/options.java/nbproject/project.xml | 1 + java/java.file.launcher/nbproject/project.xml | 25 ++++ .../AttributeBasedSingleFileOptions.java | 6 +- .../java/file/launcher/Bundle.properties | 3 + .../GlobalSettingsOptionsPanelController.java | 101 ++++++++++++++++ .../file/launcher/GlobalSettingsPanel.form | 105 +++++++++++++++++ .../file/launcher/GlobalSettingsPanel.java | 111 ++++++++++++++++++ .../file/launcher/SingleSourceFileUtil.java | 9 ++ .../java/file/launcher/actions/JPDAStart.java | 11 +- .../file/launcher/actions/LaunchProcess.java | 3 + .../SingleJavaSourceRunActionProvider.java | 54 ++++++--- .../netbeans/modules/java/Bundle.properties | 2 + .../org/netbeans/modules/java/JavaNode.java | 16 +++ 13 files changed, 424 insertions(+), 23 deletions(-) create mode 100644 java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsOptionsPanelController.java create mode 100644 java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.form create mode 100644 java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.java diff --git a/extide/options.java/nbproject/project.xml b/extide/options.java/nbproject/project.xml index f3e816385eaa..922c247c3044 100644 --- a/extide/options.java/nbproject/project.xml +++ b/extide/options.java/nbproject/project.xml @@ -63,6 +63,7 @@ org.netbeans.modules.profiler.options org.netbeans.modules.vmd.componentssupport org.netbeans.modules.jshell.support + org.netbeans.modules.java.file.launcher org.netbeans.modules.options.java.api diff --git a/java/java.file.launcher/nbproject/project.xml b/java/java.file.launcher/nbproject/project.xml index 195981d25ec1..23476f52a29c 100644 --- a/java/java.file.launcher/nbproject/project.xml +++ b/java/java.file.launcher/nbproject/project.xml @@ -140,6 +140,23 @@ 1.84 + + org.netbeans.modules.options.api + + + + 1 + 1.71 + + + + org.netbeans.modules.options.java + + + + 1.39 + + org.netbeans.modules.parsing.api @@ -184,6 +201,14 @@ 1.65 + + org.openide.awt + + + + 7.94 + + org.openide.filesystems diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/AttributeBasedSingleFileOptions.java b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/AttributeBasedSingleFileOptions.java index 3312081635cf..550ff2e03d00 100644 --- a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/AttributeBasedSingleFileOptions.java +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/AttributeBasedSingleFileOptions.java @@ -20,6 +20,7 @@ import java.net.URI; import javax.swing.event.ChangeListener; +import org.netbeans.api.java.platform.JavaPlatformManager; import org.netbeans.modules.java.file.launcher.queries.MultiSourceRootProvider; import org.netbeans.modules.java.file.launcher.spi.SingleFileOptionsQueryImplementation; import org.openide.filesystems.FileAttributeEvent; @@ -29,6 +30,7 @@ import org.openide.filesystems.FileUtil; import org.openide.util.ChangeSupport; import org.openide.util.Lookup; +import org.openide.util.NbPreferences; import org.openide.util.RequestProcessor; import org.openide.util.WeakListeners; import org.openide.util.lookup.ServiceProvider; @@ -104,7 +106,9 @@ public String getOptions() { vmOptionsObj = root != null ? root.getAttribute(SingleSourceFileUtil.FILE_VM_OPTIONS) : null; - return vmOptionsObj != null ? (String) vmOptionsObj : ""; + String globalVmOptions = NbPreferences.forModule(JavaPlatformManager.class).get(SingleSourceFileUtil.GLOBAL_VM_OPTIONS, ""); // NOI18N + + return vmOptionsObj != null ? (String) vmOptionsObj + " " + globalVmOptions : globalVmOptions; // NOI18N } @Override diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/Bundle.properties b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/Bundle.properties index e091b91b1845..680d623800aa 100644 --- a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/Bundle.properties +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/Bundle.properties @@ -16,3 +16,6 @@ # under the License. OpenIDE-Module-Name=Java File Launcher +GlobalSettingsPanel.vmLabel.text=Additional VM Options: +GlobalSettingsPanel.singleRunCheckBox.text=Stop before Run +GlobalSettingsPanel.jLabel1.text=Execution of standalone Java sources: diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsOptionsPanelController.java b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsOptionsPanelController.java new file mode 100644 index 000000000000..784b88949d4f --- /dev/null +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsOptionsPanelController.java @@ -0,0 +1,101 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ +package org.netbeans.modules.java.file.launcher; + +import java.beans.PropertyChangeListener; +import java.beans.PropertyChangeSupport; +import javax.swing.JComponent; +import javax.swing.SwingUtilities; +import org.netbeans.spi.options.OptionsPanelController; +import org.netbeans.modules.options.java.api.JavaOptions; +import org.openide.util.HelpCtx; +import org.openide.util.Lookup; + +@OptionsPanelController.SubRegistration( + location = JavaOptions.JAVA, + displayName = "#AdvancedOption_DisplayName_GlobalSettings", + keywords = "#AdvancedOption_Keywords_GlobalSettings", + keywordsCategory = "Java/GlobalSettings" +) +@org.openide.util.NbBundle.Messages({"AdvancedOption_DisplayName_GlobalSettings=Source Launcher", "AdvancedOption_Keywords_GlobalSettings=single source"}) +public final class GlobalSettingsOptionsPanelController extends OptionsPanelController { + + private GlobalSettingsPanel panel; + private final PropertyChangeSupport pcs = new PropertyChangeSupport(this); + private boolean changed; + + public void update() { + getPanel().load(); + changed = false; + } + + public void applyChanges() { + SwingUtilities.invokeLater(new Runnable() { + @Override + public void run() { + getPanel().store(); + changed = false; + } + }); + } + + public void cancel() { + // need not do anything special, if no changes have been persisted yet + } + + public boolean isValid() { + return getPanel().valid(); + } + + public boolean isChanged() { + return changed; + } + + public HelpCtx getHelpCtx() { + return null; // new HelpCtx("...ID") if you have a help set + } + + public JComponent getComponent(Lookup masterLookup) { + return getPanel(); + } + + public void addPropertyChangeListener(PropertyChangeListener l) { + pcs.addPropertyChangeListener(l); + } + + public void removePropertyChangeListener(PropertyChangeListener l) { + pcs.removePropertyChangeListener(l); + } + + private GlobalSettingsPanel getPanel() { + if (panel == null) { + panel = new GlobalSettingsPanel(this); + } + return panel; + } + + void changed() { + if (!changed) { + changed = true; + pcs.firePropertyChange(OptionsPanelController.PROP_CHANGED, false, true); + } + pcs.firePropertyChange(OptionsPanelController.PROP_VALID, null, null); + } + +} diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.form b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.form new file mode 100644 index 000000000000..b96e849c1781 --- /dev/null +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.form @@ -0,0 +1,105 @@ + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.java b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.java new file mode 100644 index 000000000000..d61fe085a84f --- /dev/null +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/GlobalSettingsPanel.java @@ -0,0 +1,111 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ +package org.netbeans.modules.java.file.launcher; + +import java.util.prefs.Preferences; +import org.netbeans.api.java.platform.JavaPlatformManager; +import org.openide.util.NbPreferences; + +final class GlobalSettingsPanel extends javax.swing.JPanel { + + private final GlobalSettingsOptionsPanelController controller; + + GlobalSettingsPanel(GlobalSettingsOptionsPanelController controller) { + this.controller = controller; + initComponents(); + // TODO listen to changes in form fields and call controller.changed() + } + + /** + * This method is called from within the constructor to initialize the form. + * WARNING: Do NOT modify this code. The content of this method is always + * regenerated by the Form Editor. + */ + // //GEN-BEGIN:initComponents + private void initComponents() { + + vmTextField = new javax.swing.JTextField(); + vmLabel = new javax.swing.JLabel(); + singleRunCheckBox = new javax.swing.JCheckBox(); + jLabel1 = new javax.swing.JLabel(); + + vmLabel.setLabelFor(vmTextField); + org.openide.awt.Mnemonics.setLocalizedText(vmLabel, org.openide.util.NbBundle.getMessage(GlobalSettingsPanel.class, "GlobalSettingsPanel.vmLabel.text")); // NOI18N + + org.openide.awt.Mnemonics.setLocalizedText(singleRunCheckBox, org.openide.util.NbBundle.getMessage(GlobalSettingsPanel.class, "GlobalSettingsPanel.singleRunCheckBox.text")); // NOI18N + + org.openide.awt.Mnemonics.setLocalizedText(jLabel1, org.openide.util.NbBundle.getMessage(GlobalSettingsPanel.class, "GlobalSettingsPanel.jLabel1.text")); // NOI18N + + javax.swing.GroupLayout layout = new javax.swing.GroupLayout(this); + this.setLayout(layout); + layout.setHorizontalGroup( + layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING) + .addGroup(layout.createSequentialGroup() + .addContainerGap() + .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING) + .addGroup(layout.createSequentialGroup() + .addComponent(vmLabel) + .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.UNRELATED) + .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING) + .addComponent(singleRunCheckBox, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE) + .addComponent(vmTextField))) + .addGroup(layout.createSequentialGroup() + .addComponent(jLabel1) + .addGap(0, 0, Short.MAX_VALUE))) + .addContainerGap()) + ); + layout.setVerticalGroup( + layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING) + .addGroup(layout.createSequentialGroup() + .addContainerGap() + .addComponent(jLabel1) + .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED) + .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING) + .addComponent(vmLabel) + .addComponent(vmTextField, javax.swing.GroupLayout.PREFERRED_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.PREFERRED_SIZE)) + .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED) + .addComponent(singleRunCheckBox) + .addContainerGap(javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)) + ); + }// //GEN-END:initComponents + + void load() { + Preferences prefs = NbPreferences.forModule(JavaPlatformManager.class); + vmTextField.setText(prefs.get(SingleSourceFileUtil.GLOBAL_VM_OPTIONS, "")); // NOI18N + singleRunCheckBox.setSelected(prefs.getBoolean(SingleSourceFileUtil.GLOBAL_STOP_AND_RUN_OPTION, false)); // NOI18N + } + + void store() { + Preferences prefs = NbPreferences.forModule(JavaPlatformManager.class); + prefs.put(SingleSourceFileUtil.GLOBAL_VM_OPTIONS, vmTextField.getText()); // NOI18N + prefs.putBoolean(SingleSourceFileUtil.GLOBAL_STOP_AND_RUN_OPTION, singleRunCheckBox.isSelected()); // NOI18N + } + + boolean valid() { + // TODO check whether form is consistent and complete + return true; + } + + // Variables declaration - do not modify//GEN-BEGIN:variables + private javax.swing.JLabel jLabel1; + private javax.swing.JCheckBox singleRunCheckBox; + private javax.swing.JLabel vmLabel; + private javax.swing.JTextField vmTextField; + // End of variables declaration//GEN-END:variables +} diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/SingleSourceFileUtil.java b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/SingleSourceFileUtil.java index d8fb70bb990a..662bfefb4ab6 100644 --- a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/SingleSourceFileUtil.java +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/SingleSourceFileUtil.java @@ -31,6 +31,7 @@ import javax.swing.event.ChangeEvent; import javax.swing.event.ChangeListener; import org.netbeans.api.java.platform.JavaPlatform; +import org.netbeans.api.java.platform.JavaPlatformManager; import org.netbeans.api.project.FileOwnerQuery; import org.netbeans.modules.java.file.launcher.queries.MultiSourceRootProvider; import org.netbeans.modules.java.file.launcher.spi.SingleFileOptionsQueryImplementation; @@ -41,6 +42,7 @@ import org.openide.loaders.DataObject; import org.openide.util.ChangeSupport; import org.openide.util.Lookup; +import org.openide.util.NbPreferences; /** * @@ -60,6 +62,9 @@ public static int findJavaVersion() throws NumberFormatException { return version; } + public static final String GLOBAL_VM_OPTIONS = "java_file_launcher_global_vm_options"; //NOI18N + public static final String GLOBAL_STOP_AND_RUN_OPTION = "java_file_launcher_global_stop_and_run_option"; //NOI18N + // synced with JavaNode public static final String FILE_ARGUMENTS = "single_file_run_arguments"; //NOI18N public static final String FILE_JDK = "single_file_run_jdk"; //NOI18N @@ -117,6 +122,10 @@ public static Process compileJavaSource(FileObject fileObject, JavaPlatform jdk) if (!vmOptions.isEmpty()) { compileCommandList.addAll(Arrays.asList(vmOptions.split(" "))); //NOI18N } + String globalVmOptions = NbPreferences.forModule(JavaPlatformManager.class).get(GLOBAL_VM_OPTIONS, "").trim(); // NOI18N + if (!globalVmOptions.isEmpty()) { + compileCommandList.addAll(Arrays.asList(globalVmOptions.split(" "))); //NOI18N + } compileCommandList.add(fileObject.getPath()); ProcessBuilder compileProcessBuilder = new ProcessBuilder(compileCommandList); compileProcessBuilder.directory(new File(fileObject.getParent().getPath())); diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/JPDAStart.java b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/JPDAStart.java index ede73c49dfe4..329b54428eac 100644 --- a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/JPDAStart.java +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/JPDAStart.java @@ -39,7 +39,6 @@ import org.netbeans.api.debugger.jpda.JPDADebugger; import org.netbeans.modules.java.file.launcher.SingleSourceFileUtil; import org.netbeans.spi.java.classpath.support.ClassPathSupport; -import org.openide.windows.InputOutput; /** * Start the JPDA debugger. @@ -52,11 +51,9 @@ public class JPDAStart implements Runnable { private static final String TRANSPORT = "dt_socket"; //NOI18N private final Object[] lock = new Object[2]; - private final InputOutput io; private final FileObject fileObject; - JPDAStart(InputOutput inputOutput, FileObject fileObject) { - io = inputOutput; + JPDAStart(FileObject fileObject) { this.fileObject = fileObject; } @@ -123,16 +120,16 @@ public void run() { JPDADebugger.startListening(flc, args, new Object[]{properties}); } catch (DebuggerStartException ex) { - io.getErr().println("Debugger Start Error."); //NOI18N +// io.getErr().println("Debugger Start Error."); //NOI18N SingleSourceFileUtil.LOG.log(Level.SEVERE, "Debugger Start Error.", ex); } }); } catch (java.io.IOException ioex) { - io.getErr().println("IO Error:"); //NOI18N +// io.getErr().println("IO Error:"); //NOI18N // org.openide.ErrorManager.getDefault().notify(ioex); lock[1] = ioex; } catch (com.sun.jdi.connect.IllegalConnectorArgumentsException icaex) { - io.getErr().println("Illegal Connector"); //NOI18N +// io.getErr().println("Illegal Connector"); //NOI18N lock[1] = icaex; } finally { lock.notify(); diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/LaunchProcess.java b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/LaunchProcess.java index befa4cf8c89f..daf1bbdddf38 100644 --- a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/LaunchProcess.java +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/LaunchProcess.java @@ -34,6 +34,7 @@ import org.openide.filesystems.FileObject; import org.openide.filesystems.FileUtil; import org.openide.util.BaseUtilities; +import org.openide.util.NbPreferences; import org.openide.util.Utilities; final class LaunchProcess implements Callable { @@ -82,6 +83,8 @@ private Process setupProcess(String port) throws InterruptedException { ExplicitProcessParameters.builder() .args(readArgumentsFromAttribute(fileObject, SingleSourceFileUtil.FILE_ARGUMENTS)) .launcherArgs(readArgumentsFromAttribute(fileObject, SingleSourceFileUtil.FILE_VM_OPTIONS)) + .launcherArgs(Arrays.asList(BaseUtilities.parseParameters( + NbPreferences.forModule(JavaPlatformManager.class).get(SingleSourceFileUtil.GLOBAL_VM_OPTIONS, "").trim()))) //NOI18N .workingDirectory(workDir) .build(); diff --git a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/SingleJavaSourceRunActionProvider.java b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/SingleJavaSourceRunActionProvider.java index 007764c96487..05c9d1ff1ccd 100644 --- a/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/SingleJavaSourceRunActionProvider.java +++ b/java/java.file.launcher/src/org/netbeans/modules/java/file/launcher/actions/SingleJavaSourceRunActionProvider.java @@ -18,19 +18,22 @@ */ package org.netbeans.modules.java.file.launcher.actions; +import java.nio.charset.Charset; +import java.util.Map; +import java.util.WeakHashMap; import java.util.concurrent.Future; import org.netbeans.api.extexecution.ExecutionDescriptor; import org.netbeans.api.extexecution.ExecutionService; import org.netbeans.modules.java.file.launcher.SingleSourceFileUtil; import org.netbeans.api.extexecution.base.ExplicitProcessParameters; -import org.netbeans.spi.project.ActionProgress; +import org.netbeans.api.java.platform.JavaPlatformManager; import org.netbeans.spi.project.ActionProvider; +import org.openide.LifecycleManager; import org.openide.filesystems.FileObject; import org.openide.util.Lookup; import org.openide.util.NbBundle; +import org.openide.util.NbPreferences; import org.openide.util.lookup.ServiceProvider; -import org.openide.windows.IOProvider; -import org.openide.windows.InputOutput; /** * This class provides support to run a single Java file without a parent @@ -40,6 +43,10 @@ */ @ServiceProvider(service = ActionProvider.class) public final class SingleJavaSourceRunActionProvider implements ActionProvider { + + private final Map> running = new WeakHashMap<>(); + private volatile boolean rerun = false; + @Override public String[] getSupportedActions() { return new String[]{ @@ -53,26 +60,43 @@ public String[] getSupportedActions() { }) @Override public void invokeAction(String command, Lookup context) throws IllegalArgumentException { + LifecycleManager.getDefault().saveAll(); FileObject fileObject = SingleSourceFileUtil.getJavaFileWithoutProjectFromLookup(context); - if (fileObject == null) + if (fileObject == null) return; + var previous = running.get(fileObject); + if (previous != null && !previous.isDone()) { + rerun = true; + if (previous.cancel(true)) { + return; + } + rerun = false; + } ExplicitProcessParameters params = ExplicitProcessParameters.buildExplicitParameters(context); - InputOutput io = IOProvider.getDefault().getIO(Bundle.CTL_SingleJavaFile(), false); - ActionProgress progress = ActionProgress.start(context); + String preferredEncoding = System.getProperty("native.encoding"); // NOI18N ExecutionDescriptor descriptor = new ExecutionDescriptor(). + showProgress(true). controllable(true). frontWindow(true). preExecution(null). - inputOutput(io). + inputVisible(true). + charset(preferredEncoding != null ? Charset.forName(preferredEncoding) : null). postExecution((exitCode) -> { - progress.finished(exitCode == 0); + if (rerun) { + rerun = false; + invokeAction(command, context); + } }); - LaunchProcess process = invokeActionHelper(io, command, fileObject, params); + LaunchProcess process = invokeActionHelper(command, fileObject, params); ExecutionService exeService = ExecutionService.newService( process, - descriptor, "Running Single Java File"); - Future exitCode = exeService.run(); + descriptor, fileObject.getNameExt()); + + Future future = exeService.run(); + if (NbPreferences.forModule(JavaPlatformManager.class).getBoolean(SingleSourceFileUtil.GLOBAL_STOP_AND_RUN_OPTION, false)) { + running.put(fileObject, future); + } } @Override @@ -80,11 +104,11 @@ public boolean isActionEnabled(String command, Lookup context) throws IllegalArg FileObject fileObject = SingleSourceFileUtil.getJavaFileWithoutProjectFromLookup(context); return fileObject != null; } - - final LaunchProcess invokeActionHelper (InputOutput io, String command, FileObject fo, ExplicitProcessParameters params) { + + final LaunchProcess invokeActionHelper (String command, FileObject fo, ExplicitProcessParameters params) { JPDAStart start = ActionProvider.COMMAND_DEBUG_SINGLE.equals(command) ? - new JPDAStart(io, fo) : null; + new JPDAStart(fo) : null; return new LaunchProcess(fo, start, params); } - + } diff --git a/java/java.source/src/org/netbeans/modules/java/Bundle.properties b/java/java.source/src/org/netbeans/modules/java/Bundle.properties index 1d3268d9067e..e5ea1acd023d 100644 --- a/java/java.source/src/org/netbeans/modules/java/Bundle.properties +++ b/java/java.source/src/org/netbeans/modules/java/Bundle.properties @@ -37,6 +37,8 @@ PROP_JavaNode_singlefile_arguments=Program Arguments HINT_JavaNode_singlefile_arguments=Arguments passed to the main method while running the file. PROP_JavaNode_singlefile_options=VM Options HINT_JavaNode_singlefile_options=VM Options to be considered while running the file. +PROP_JavaNode_singlefile_global_options=Additional VM Options +HINT_JavaNode_singlefile_global_options=Additional VM Options configured in Settings... / Java / Source Launcher. PROP_JavaNode_singlefile_registerRoot=Enable Source File Launcher Indexing HINT_JavaNode_singlefile_registerRoot=The root corresponding to this file should have source file launcher indexing enabled PROP_JavaNode_classfile_version=Classfile Version diff --git a/java/java.source/src/org/netbeans/modules/java/JavaNode.java b/java/java.source/src/org/netbeans/modules/java/JavaNode.java index 60bfeced4598..51089d2cceb3 100644 --- a/java/java.source/src/org/netbeans/modules/java/JavaNode.java +++ b/java/java.source/src/org/netbeans/modules/java/JavaNode.java @@ -82,6 +82,7 @@ import org.openide.util.Exceptions; import org.openide.util.ImageUtilities; import org.openide.util.Lookup; +import org.openide.util.NbPreferences; import org.openide.util.RequestProcessor; import org.openide.util.Utilities; import org.openide.util.WeakListeners; @@ -108,6 +109,7 @@ public final class JavaNode extends DataNode implements ChangeListener { private static final String EXECUTABLE_BADGE_URL = "org/netbeans/modules/java/resources/executable-badge.png"; //NOI18N private static final String NEEDS_COMPILE_BADGE_URL = "org/netbeans/modules/java/resources/needs-compile.png"; //NOI18N // synced with org.netbeans.modules.java.file.launcher.SingleSourceFileUtil + private static final String GLOBAL_VM_OPTIONS = "java_file_launcher_global_vm_options"; //NOI18N private static final String FILE_ARGUMENTS = "single_file_run_arguments"; //NOI18N private static final String FILE_JDK = "single_file_run_jdk"; //NOI18N private static final String FILE_VM_OPTIONS = "single_file_vm_options"; //NOI18N @@ -247,6 +249,7 @@ protected final Sheet createSheet () { ss.put(new RunFileJDKProperty(dObj)); ss.put(new JavaFileAttributeProperty(dObj, FILE_ARGUMENTS, "runFileArguments", "singlefile_arguments")); // NOI18N ss.put(new JavaFileAttributeProperty(dObj, FILE_VM_OPTIONS, "runFileVMOptions", "singlefile_options")); // NOI18N + ss.put(new JavaFileGlobalOptionsProperty()); sheet.put(ss); } @@ -522,6 +525,19 @@ public void setValue(String o) { } } + // read-only global VM options + private static final class JavaFileGlobalOptionsProperty extends PropertySupport.ReadOnly { + + public JavaFileGlobalOptionsProperty() { + super("runFileGlobalOptions", String.class, getMessage(JavaNode.class, "PROP_JavaNode_singlefile_global_options"), getMessage(JavaNode.class, "HINT_JavaNode_singlefile_global_options")); // NOI18N + } + + @Override + public String getValue() { + return NbPreferences.forModule(JavaPlatformManager.class).get(GLOBAL_VM_OPTIONS, "").trim(); + } + } + @Override public void stateChanged(ChangeEvent e) { WORKER.post(new BuildStatusTask(this));