mirror of
				https://github.com/DanilaFe/abacus
				synced 2025-11-04 02:43:41 -08:00 
			
		
		
		
	Merge pull request #52 from DanilaFe/definition-files
Allow loading definition files.
This commit is contained in:
		
						commit
						4188c83a66
					
				@ -63,13 +63,13 @@ class Abacus(val configuration: Configuration) {
 | 
				
			|||||||
     * Reloads the Abacus core.
 | 
					     * Reloads the Abacus core.
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    fun reload(){
 | 
					    fun reload(){
 | 
				
			||||||
        pluginManager.reload()
 | 
					        with(mutableContext){
 | 
				
			||||||
        with(mutableContext) {
 | 
					 | 
				
			||||||
            numberImplementation = pluginManager.numberImplementationFor(configuration.numberImplementation)
 | 
					 | 
				
			||||||
                    ?: StandardPlugin.IMPLEMENTATION_NAIVE
 | 
					 | 
				
			||||||
            clearVariables()
 | 
					            clearVariables()
 | 
				
			||||||
            clearDefinitions()
 | 
					            clearDefinitions()
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        pluginManager.reload()
 | 
				
			||||||
 | 
					        mutableContext.numberImplementation = pluginManager.numberImplementationFor(configuration.numberImplementation)
 | 
				
			||||||
 | 
					                ?: StandardPlugin.IMPLEMENTATION_NAIVE
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * Merges the current context with the provided one, updating
 | 
					     * Merges the current context with the provided one, updating
 | 
				
			||||||
 | 
				
			|||||||
@ -26,7 +26,7 @@ public class AbacusApplication extends Application {
 | 
				
			|||||||
        FXMLLoader loader = new FXMLLoader(getClass().getResource("/abacus.fxml"));
 | 
					        FXMLLoader loader = new FXMLLoader(getClass().getResource("/abacus.fxml"));
 | 
				
			||||||
        Parent parent = loader.load();
 | 
					        Parent parent = loader.load();
 | 
				
			||||||
        controller = loader.getController();
 | 
					        controller = loader.getController();
 | 
				
			||||||
        Scene mainScene = new Scene(parent, 320, 480);
 | 
					        Scene mainScene = new Scene(parent, 420, 520);
 | 
				
			||||||
        primaryStage.setScene(mainScene);
 | 
					        primaryStage.setScene(mainScene);
 | 
				
			||||||
        primaryStage.setTitle("Abacus");
 | 
					        primaryStage.setTitle("Abacus");
 | 
				
			||||||
        primaryStage.show();
 | 
					        primaryStage.show();
 | 
				
			||||||
 | 
				
			|||||||
@ -8,6 +8,7 @@ import javafx.fxml.FXML;
 | 
				
			|||||||
import javafx.scene.control.*;
 | 
					import javafx.scene.control.*;
 | 
				
			||||||
import javafx.scene.control.cell.CheckBoxListCell;
 | 
					import javafx.scene.control.cell.CheckBoxListCell;
 | 
				
			||||||
import javafx.scene.text.Text;
 | 
					import javafx.scene.text.Text;
 | 
				
			||||||
 | 
					import javafx.stage.FileChooser;
 | 
				
			||||||
import javafx.util.Callback;
 | 
					import javafx.util.Callback;
 | 
				
			||||||
import javafx.util.StringConverter;
 | 
					import javafx.util.StringConverter;
 | 
				
			||||||
import org.nwapw.abacus.Abacus;
 | 
					import org.nwapw.abacus.Abacus;
 | 
				
			||||||
@ -23,9 +24,12 @@ import org.nwapw.abacus.EvaluationResult;
 | 
				
			|||||||
import org.nwapw.abacus.tree.nodes.TreeNode;
 | 
					import org.nwapw.abacus.tree.nodes.TreeNode;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import java.io.File;
 | 
					import java.io.File;
 | 
				
			||||||
 | 
					import java.io.FileNotFoundException;
 | 
				
			||||||
 | 
					import java.io.FileReader;
 | 
				
			||||||
import java.io.IOException;
 | 
					import java.io.IOException;
 | 
				
			||||||
import java.util.ArrayList;
 | 
					import java.util.ArrayList;
 | 
				
			||||||
import java.util.Comparator;
 | 
					import java.util.Comparator;
 | 
				
			||||||
 | 
					import java.util.Scanner;
 | 
				
			||||||
import java.util.Set;
 | 
					import java.util.Set;
 | 
				
			||||||
import java.util.stream.Collectors;
 | 
					import java.util.stream.Collectors;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -69,6 +73,7 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
     * Constant string that is displayed if the calculations are interrupted by an exception.
 | 
					     * Constant string that is displayed if the calculations are interrupted by an exception.
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    private static final String ERR_EXCEPTION = "Exception Thrown";
 | 
					    private static final String ERR_EXCEPTION = "Exception Thrown";
 | 
				
			||||||
 | 
					    private static final String ERR_DEFINITION = "Definition Error";
 | 
				
			||||||
    @FXML
 | 
					    @FXML
 | 
				
			||||||
    private TabPane coreTabPane;
 | 
					    private TabPane coreTabPane;
 | 
				
			||||||
    @FXML
 | 
					    @FXML
 | 
				
			||||||
@ -103,6 +108,8 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
    private ListView<Documentation> functionListView;
 | 
					    private ListView<Documentation> functionListView;
 | 
				
			||||||
    @FXML
 | 
					    @FXML
 | 
				
			||||||
    private TextField functionListSearchField;
 | 
					    private TextField functionListSearchField;
 | 
				
			||||||
 | 
					    @FXML
 | 
				
			||||||
 | 
					    private ListView<String> definitionFilesView;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * The list of history entries, created by the users.
 | 
					     * The list of history entries, created by the users.
 | 
				
			||||||
@ -128,6 +135,10 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
     * The filtered list displayed to the user.
 | 
					     * The filtered list displayed to the user.
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    private FilteredList<Documentation> functionFilter;
 | 
					    private FilteredList<Documentation> functionFilter;
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * The list of definition files to be loaded.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    private ObservableList<String> definitionFiles;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * The abacus instance used for changing the plugin configuration.
 | 
					     * The abacus instance used for changing the plugin configuration.
 | 
				
			||||||
@ -251,6 +262,9 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
            if (oldValue.equals(settingsTab)) alertIfApplyNeeded(true);
 | 
					            if (oldValue.equals(settingsTab)) alertIfApplyNeeded(true);
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        definitionFiles = FXCollections.observableArrayList();
 | 
				
			||||||
 | 
					        definitionFilesView.setItems(definitionFiles);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        abacus = new Abacus(new ExtendedConfiguration(CONFIG_FILE));
 | 
					        abacus = new Abacus(new ExtendedConfiguration(CONFIG_FILE));
 | 
				
			||||||
        PluginManager abacusPluginManager = abacus.getPluginManager();
 | 
					        PluginManager abacusPluginManager = abacus.getPluginManager();
 | 
				
			||||||
        abacusPluginManager.addListener(this);
 | 
					        abacusPluginManager.addListener(this);
 | 
				
			||||||
@ -304,6 +318,23 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
        reloadAlertShown = false;
 | 
					        reloadAlertShown = false;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private void loadDefinitionFile(String fileName){
 | 
				
			||||||
 | 
					        File definitionFile = new File(fileName);
 | 
				
			||||||
 | 
					        if(!definitionFile.exists()) return;
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            FileReader fileReader = new FileReader(definitionFile);
 | 
				
			||||||
 | 
					            Scanner scanner = new Scanner(fileReader);
 | 
				
			||||||
 | 
					            while(scanner.hasNext()){
 | 
				
			||||||
 | 
					                EvaluationResult result = abacus.evaluateTree(abacus.parseString(scanner.nextLine()));
 | 
				
			||||||
 | 
					                abacus.applyToContext(result.getResultingContext());
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        } catch (AbacusException abacusError) {
 | 
				
			||||||
 | 
					            outputText.setText(ERR_DEFINITION + "(" + abacusError.getMessage() + ")");
 | 
				
			||||||
 | 
					        } catch (RuntimeException runtime) {
 | 
				
			||||||
 | 
					            outputText.setText(ERR_DEFINITION + "(" + ERR_EXCEPTION + ")");
 | 
				
			||||||
 | 
					        } catch (FileNotFoundException ignored) {}
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @FXML
 | 
					    @FXML
 | 
				
			||||||
    public void performScan() {
 | 
					    public void performScan() {
 | 
				
			||||||
        PluginManager abacusPluginManager = abacus.getPluginManager();
 | 
					        PluginManager abacusPluginManager = abacus.getPluginManager();
 | 
				
			||||||
@ -325,23 +356,44 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    @FXML
 | 
					    @FXML
 | 
				
			||||||
    public void performSave() {
 | 
					    public void performSave() {
 | 
				
			||||||
        Configuration configuration = abacus.getConfiguration();
 | 
					        ExtendedConfiguration configuration = (ExtendedConfiguration) abacus.getConfiguration();
 | 
				
			||||||
        configuration.setNumberImplementation(numberImplementationBox.getSelectionModel().getSelectedItem());
 | 
					        configuration.setNumberImplementation(numberImplementationBox.getSelectionModel().getSelectedItem());
 | 
				
			||||||
        Set<String> disabledPlugins = configuration.getDisabledPlugins();
 | 
					        Set<String> disabledPlugins = configuration.getDisabledPlugins();
 | 
				
			||||||
        disabledPlugins.clear();
 | 
					        disabledPlugins.clear();
 | 
				
			||||||
        for (ToggleablePlugin pluginEntry : enabledPlugins) {
 | 
					        for (ToggleablePlugin pluginEntry : enabledPlugins) {
 | 
				
			||||||
            if (!pluginEntry.isEnabled()) disabledPlugins.add(pluginEntry.getClassName());
 | 
					            if (!pluginEntry.isEnabled()) disabledPlugins.add(pluginEntry.getClassName());
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        Set<String> abacusDefinitionFiles = configuration.getDefinitionFiles();
 | 
				
			||||||
 | 
					        abacusDefinitionFiles.clear();
 | 
				
			||||||
 | 
					        abacusDefinitionFiles.addAll(definitionFiles);
 | 
				
			||||||
        if (computationLimitField.getText().matches("\\d*(\\.\\d+)?") && computationLimitField.getText().length() != 0)
 | 
					        if (computationLimitField.getText().matches("\\d*(\\.\\d+)?") && computationLimitField.getText().length() != 0)
 | 
				
			||||||
            ((ExtendedConfiguration) configuration).setComputationDelay(Double.parseDouble(computationLimitField.getText()));
 | 
					            configuration.setComputationDelay(Double.parseDouble(computationLimitField.getText()));
 | 
				
			||||||
        ((ExtendedConfiguration) configuration).saveTo(CONFIG_FILE);
 | 
					        configuration.saveTo(CONFIG_FILE);
 | 
				
			||||||
        changesMade = false;
 | 
					        changesMade = false;
 | 
				
			||||||
        reloadAlertShown = false;
 | 
					        reloadAlertShown = false;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @FXML
 | 
				
			||||||
 | 
					    public void performAddDefinitionFile(){
 | 
				
			||||||
 | 
					        FileChooser fileChooser = new FileChooser();
 | 
				
			||||||
 | 
					        fileChooser.setTitle("Add definition file");
 | 
				
			||||||
 | 
					        File selectedFile = fileChooser.showOpenDialog(null);
 | 
				
			||||||
 | 
					        if(selectedFile == null) return;
 | 
				
			||||||
 | 
					        String absolutePath = selectedFile.getAbsolutePath();
 | 
				
			||||||
 | 
					        if(!definitionFiles.contains(absolutePath)) definitionFiles.add(absolutePath);
 | 
				
			||||||
 | 
					        changesMade = true;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @FXML
 | 
				
			||||||
 | 
					    public void performRemoveDefinitionFile(){
 | 
				
			||||||
 | 
					        String selectedItem = definitionFilesView.getSelectionModel().getSelectedItem();
 | 
				
			||||||
 | 
					        if(selectedItem != null) definitionFiles.remove(selectedItem);
 | 
				
			||||||
 | 
					        changesMade = true;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Override
 | 
					    @Override
 | 
				
			||||||
    public void onLoad(PluginManager manager) {
 | 
					    public void onLoad(PluginManager manager) {
 | 
				
			||||||
        Configuration configuration = abacus.getConfiguration();
 | 
					        ExtendedConfiguration configuration = (ExtendedConfiguration) abacus.getConfiguration();
 | 
				
			||||||
        Set<String> disabledPlugins = configuration.getDisabledPlugins();
 | 
					        Set<String> disabledPlugins = configuration.getDisabledPlugins();
 | 
				
			||||||
        numberImplementationOptions.addAll(abacus.getPluginManager().getAllNumberImplementations());
 | 
					        numberImplementationOptions.addAll(abacus.getPluginManager().getAllNumberImplementations());
 | 
				
			||||||
        String actualImplementation = configuration.getNumberImplementation();
 | 
					        String actualImplementation = configuration.getNumberImplementation();
 | 
				
			||||||
@ -367,6 +419,10 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
            return documentationInstance;
 | 
					            return documentationInstance;
 | 
				
			||||||
        }).collect(Collectors.toCollection(ArrayList::new)));
 | 
					        }).collect(Collectors.toCollection(ArrayList::new)));
 | 
				
			||||||
        functionList.sort(Comparator.comparing(Documentation::getCodeName));
 | 
					        functionList.sort(Comparator.comparing(Documentation::getCodeName));
 | 
				
			||||||
 | 
					        definitionFiles.addAll(configuration.getDefinitionFiles());
 | 
				
			||||||
 | 
					        for(String file : definitionFiles){
 | 
				
			||||||
 | 
					            loadDefinitionFile(file);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Override
 | 
					    @Override
 | 
				
			||||||
@ -374,6 +430,7 @@ public class AbacusController implements PluginListener {
 | 
				
			|||||||
        functionList.clear();
 | 
					        functionList.clear();
 | 
				
			||||||
        enabledPlugins.clear();
 | 
					        enabledPlugins.clear();
 | 
				
			||||||
        numberImplementationOptions.clear();
 | 
					        numberImplementationOptions.clear();
 | 
				
			||||||
 | 
					        definitionFiles.clear();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -17,8 +17,9 @@ import java.io.File
 | 
				
			|||||||
 * @param disabledPlugins the list of plugins that should be disabled, same as [Configuration.disabledPlugins]
 | 
					 * @param disabledPlugins the list of plugins that should be disabled, same as [Configuration.disabledPlugins]
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
class ExtendedConfiguration(var computationDelay: Double = 0.0,
 | 
					class ExtendedConfiguration(var computationDelay: Double = 0.0,
 | 
				
			||||||
                     implementation: String = "<default>",
 | 
					                            definitionFiles: Array<String> = emptyArray(),
 | 
				
			||||||
                     disabledPlugins: Array<String> = emptyArray())
 | 
					                            implementation: String = "<default>",
 | 
				
			||||||
 | 
					                            disabledPlugins: Array<String> = emptyArray())
 | 
				
			||||||
    : Configuration(implementation, disabledPlugins) {
 | 
					    : Configuration(implementation, disabledPlugins) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    companion object {
 | 
					    companion object {
 | 
				
			||||||
@ -27,6 +28,7 @@ class ExtendedConfiguration(var computationDelay: Double = 0.0,
 | 
				
			|||||||
         */
 | 
					         */
 | 
				
			||||||
        val DEFAULT_TOML_STRING = """
 | 
					        val DEFAULT_TOML_STRING = """
 | 
				
			||||||
            computationDelay=0.0
 | 
					            computationDelay=0.0
 | 
				
			||||||
 | 
					            definitionFiles=[]
 | 
				
			||||||
            implementation="naive"
 | 
					            implementation="naive"
 | 
				
			||||||
            disabledPlugins=[]
 | 
					            disabledPlugins=[]
 | 
				
			||||||
            """
 | 
					            """
 | 
				
			||||||
@ -40,6 +42,11 @@ class ExtendedConfiguration(var computationDelay: Double = 0.0,
 | 
				
			|||||||
        val DEFAULT_TOML_WRITER = TomlWriter()
 | 
					        val DEFAULT_TOML_WRITER = TomlWriter()
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * The set of files that definitions should be loaded from.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    val definitionFiles: MutableSet<String> = mutableSetOf(*definitionFiles)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * Constructs a new configuration from a file on disk.
 | 
					     * Constructs a new configuration from a file on disk.
 | 
				
			||||||
     * @param tomlFile the file from disk to load.
 | 
					     * @param tomlFile the file from disk to load.
 | 
				
			||||||
@ -59,6 +66,8 @@ class ExtendedConfiguration(var computationDelay: Double = 0.0,
 | 
				
			|||||||
        numberImplementation = config.numberImplementation
 | 
					        numberImplementation = config.numberImplementation
 | 
				
			||||||
        disabledPlugins.clear()
 | 
					        disabledPlugins.clear()
 | 
				
			||||||
        disabledPlugins.addAll(config.disabledPlugins)
 | 
					        disabledPlugins.addAll(config.disabledPlugins)
 | 
				
			||||||
 | 
					        definitionFiles.clear()
 | 
				
			||||||
 | 
					        definitionFiles.addAll(config.definitionFiles)
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
 | 
				
			|||||||
@ -47,17 +47,33 @@
 | 
				
			|||||||
                    </padding>
 | 
					                    </padding>
 | 
				
			||||||
                    <Label text="Number Implementation" GridPane.columnIndex="0" GridPane.rowIndex="0"/>
 | 
					                    <Label text="Number Implementation" GridPane.columnIndex="0" GridPane.rowIndex="0"/>
 | 
				
			||||||
                    <ComboBox fx:id="numberImplementationBox" GridPane.columnIndex="1" GridPane.rowIndex="0"/>
 | 
					                    <ComboBox fx:id="numberImplementationBox" GridPane.columnIndex="1" GridPane.rowIndex="0"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    <Label text="Plugins:" GridPane.columnIndex="0" GridPane.rowIndex="1"/>
 | 
				
			||||||
                    <ListView fx:id="enabledPluginView"
 | 
					                    <ListView fx:id="enabledPluginView"
 | 
				
			||||||
                              GridPane.rowIndex="1" GridPane.columnIndex="0"
 | 
					                              GridPane.rowIndex="2" GridPane.columnIndex="0"
 | 
				
			||||||
                              GridPane.columnSpan="2" maxHeight="100"/>
 | 
					                              GridPane.columnSpan="2" maxHeight="100"/>
 | 
				
			||||||
                    <Text GridPane.columnIndex="0" GridPane.rowIndex="2" text="Computation Limit"/>
 | 
					
 | 
				
			||||||
                    <TextField fx:id="computationLimitField" GridPane.columnIndex="1" GridPane.rowIndex="2"/>
 | 
					                    <Label GridPane.columnIndex="0" GridPane.rowIndex="3" text="Computation Limit"/>
 | 
				
			||||||
                    <FlowPane GridPane.columnIndex="0" GridPane.columnSpan="2" GridPane.rowIndex="3" hgap="10"
 | 
					                    <TextField fx:id="computationLimitField" GridPane.columnIndex="1" GridPane.rowIndex="3"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    <Label text="Definition Files:" GridPane.columnIndex="0" GridPane.rowIndex="4"/>
 | 
				
			||||||
 | 
					                    <ListView fx:id="definitionFilesView"
 | 
				
			||||||
 | 
					                              GridPane.columnIndex="0" GridPane.columnSpan="2"
 | 
				
			||||||
 | 
					                              GridPane.rowIndex="5" maxHeight="100"/>
 | 
				
			||||||
 | 
					                    <FlowPane GridPane.columnIndex="0" GridPane.columnSpan="2" GridPane.rowIndex="6" hgap="10"
 | 
				
			||||||
                              vgap="10">
 | 
					                              vgap="10">
 | 
				
			||||||
                        <Button text="Save" onAction="#performSave"/>
 | 
					                        <Button text="Add" onAction="#performAddDefinitionFile"/>
 | 
				
			||||||
                        <Button text="Reload Plugins" onAction="#performReload"/>
 | 
					                        <Button text="Remove" onAction="#performRemoveDefinitionFile"/>
 | 
				
			||||||
                        <Button text="Apply and Reload" onAction="#performSaveAndReload"/>
 | 
					                    </FlowPane>
 | 
				
			||||||
                        <Button text="Scan Plugins" onAction="#performScan"/>
 | 
					
 | 
				
			||||||
 | 
					                    <Separator GridPane.rowIndex="7"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    <FlowPane GridPane.columnIndex="0" GridPane.columnSpan="2" GridPane.rowIndex="8" hgap="10"
 | 
				
			||||||
 | 
					                              vgap="10">
 | 
				
			||||||
 | 
					                        <Button text="Save Settings" onAction="#performSave"/>
 | 
				
			||||||
 | 
					                        <Button text="Reload Plugins And Definitions" onAction="#performReload"/>
 | 
				
			||||||
 | 
					                        <Button text="Save and Reload" onAction="#performSaveAndReload"/>
 | 
				
			||||||
 | 
					                        <Button text="Reload Plugins From Disk" onAction="#performScan"/>
 | 
				
			||||||
                    </FlowPane>
 | 
					                    </FlowPane>
 | 
				
			||||||
                </GridPane>
 | 
					                </GridPane>
 | 
				
			||||||
            </Tab>
 | 
					            </Tab>
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
		Reference in New Issue
	
	Block a user