Skip to main content
Skip table of contents

Configure your tool's user interface

This section describes how to configure the way your tool is presented to the user by the Redpoint Data Management client.

Defining a custom icon

You can define a custom icon for your tool. This icon will display in the tool palette and on the canvas.

The Data Management SDK supports icons in PNG, JPG and BMP formats. Icons should be 128x128 pixels for proper display on the palette and canvas.

When naming your icon, use the unqualified class name of your tool. For example, if your class is named MyInputTool, and your icon is in PNG format, your icon should be named MyInputTool.png.

Data Management expects to find your icon in the same directory as your tool class. You may elect to keep your Java sources and resources in separate directories during development (e.g. src/main/java and src/main/resources)—that's fine, as long as they end up in the same directory after compiling and packaging. See Build and install your tool for related packaging instructions.

Setting the tool name and description

By default, Data Management will set your tool's display name to the unqualified class name of your tool. So if your class is named MyInputTool, its display name in the Palette and in the Properties pane will be MyInputTool. If you wish to show your users something more descriptive, you can override the getDisplayName() and getDescription() methods in the Tool interface to better describe your tool:

CODE
@Override
public String getDisplayName() { 
    return "My Input Tool";
}
  
@Override
public String getDescription() {
    return "This is my Input Tool. There are many like it, but this one is mine.";
}

Setting the palette category

In Data Management’s tool palette, related tools are grouped into Palette Categories. By default, any net.redpoint.dataflow.transform.Tool subclass belongs to the "Java plugin tools" category. If you wish to specify a custom category, override Tool.getPaletteCategory in your tool and specify the name of your category:

CODE
@Override
public String getPaletteCategory() { 
    return "Best. Tools. Ever.";
}

User-settable properties

Most tools require the user to perform some configuration. For example, you must specify an input file for the CSV Input tool. In the Data Management client, configuration is performed via controls in the Properties panel. Data Management will build a user interface for you by creating a user-settable property for each public data member of your class (or anything to which you've added one of the control annotations, even if the member is private). For this reason, you should only declare data members public if you want them to be user-settable properties.

The following Java types are supported by automatic user interface construction:

Java type

Control

Default annotation

Required annotation

java.lang.Boolean, boolean

Checkbox

@Checkbox

java.lang.Byte, byte
java.lang.Short, short
java.lang.Integer, int
java.lang.Long, long
java.math.BigDecimal

Spin

@Spin

java.lang.Float, float
java.lang.Double, double

Textbox

Spin

@Textbox

@Spin

org.joda.time.LocalDateTime

DateTime

@DateTime

org.joda.time.LocalDate

Date

@Date

org.joda.time.LocalTime

Time

@Time

java.lang.Long, long

Timespan picker

@TimeSpan

net.redpoint.dataflow.record.Schema

Schema editor

@SchemaEditor

java.lang.String

Single-line text edit
Multi-line text edit

@Textbox

java.lang.String

Drop list

@DropList

java.lang.String

Field selector

@FieldSelect

java.lang.String

Display-only text

@Text

net.redpoint.dataflow.io.FilePath

File browser

@FileBrowse

enum

Drop list

non-editable

@DropList

List<List<String>>

Data view (sample) grid

@DataView

net.redpoint.dataflow.types.DataType

Data type selector

@DataTypeSelect

Using other unsupported types will result in the display of an "error control" as a placeholder.

Each supported Java type has a implied "default" annotation. If you don't supply an annotation, the default annotation type will be assumed. However, if you want to alter the user interface (UI) to be something other than its simplest default behavior, or to use a different control type, you will need to supply an explicit annotation.

Some of these annotations support additional configuration elements (e.g. FileBrowse supports action, filter, and location). Review the Javadoc for details.

Using @TimeSpan

This feature is available starting in Data Management version 8.1.0.

The field annotated by @Timespan must be a long, and will contain the time span in milliseconds.

Using @DataTypeSelect

This feature is available starting in Data Management version 8.2.0.

Control annotation

The Data Management client will display a user interface control for each public data member defined in your tool, inferring the control based on the member's data type. You can customize the user interface via Java annotations defined in the net.redpoint.dataflow.annotations package. A data member can have one or more preceding annotations. By annotating a data member in this way, you can:

  • Fine-tune the behavior of specific controls (for example, setting the limits on a numeric spin control or specifying File Browse behavior).

  • Control the visibility of some controls based on the state of other controls via @Trigger and @MultiTrigger (e.g. "when I check this checkbox, show that text input").

  • Make some controls initiate actions in your code (for example, "connect to database" or "show data sample").

  • Derive the normally-static content of some controls from other data fields in your code (for example, "populate a droplist from a List<String> field").

The table above lists which annotations are applicable to each data type. The Javadoc in the net.redpoint.dataflow.annotations package provides details on each annotation's properties. You can also look at PropertiesSampleTool in the samples project, which demonstrates most of the SDK's annotation-driven user interface capabilities.

Pretty editors

You can use the @Textbox annotation with multiline=true and a syntax-highlight specification to create a pretty-editor control (for example, for XML editing):

@Textbox(caption="XML Textbox", multiline=true, syntax=Textbox.Syntax.XML)

String xmlTextbox;

This results in syntax color highlighting of XML.

Password-setting dialogs

You can use the @Textbox annotation with passwordTitle="string" attribute to control the title of a password entry control:

CODE
{{@Textbox(caption="Password", password=true, passwordTitle="Enter the password")
String password;}}

Complex controls

While most controls described in the table above are fairly easy to understand, some require additional explanation.

@DropList

The @DropList annotation defines an editable (combo box) or non-editable (select list) control.  This is perhaps best illustrated by example:

CODE
@DropList(caption="Choose your weapon", items="gun,knife,banana", editable=false)
public String weapon;

This will create a non-editable drop-list control with three items to choose from: gun, knife, or banana.

However, not all uses of this control are so simple. In many cases you want to separate the display string from the internal value string, showing one to the user while storing the other internally.

Starting with DM version 8.1, you can specify separate value and display by making each item in the list look like (display:value). For example:

CODE
@DropList(caption="Choose your weapon", items="(gun:1),(knife:2),(banana:3)", editable=false)
public String weapon;

In this case, the user will see and select from the set (gun, knife, banana), but the program will see the values (1,2,3).

Fields that are of enum type automatically become drop-lists.

Starting with version 8.2.1, you can specify a separate "display" string for enum values.  The "value" part of the specification must exactly match the enum values.

@SchemaEditor

If you annotate a Schema field (Schema being the class representing the list of fields and their types contained in Data Management records) with the @SchemaEditor annotation (or even if you don't and the Schema field is public), Data Management will produce a schema editor control in the UI. A schema editor is a complex control—it is a grid that allows the user to specify a list of fields and their types.

While this control does a lot for you, using it is incredibly simple. Just expect that the Schema field of your class will be populated with a valid Schema when configure() is called.

For an example of code that uses this control type, see the DelimitedInputSampleTool example tool code.

@Table

The @Table annotation tells Data Management to make a control that edits a list of objects.  Not just any objects, but objects whose class is a POJO annotated using Data Management annotations. For example, this code from the TableControlSampleTool.

CODE
public enum Op { UpperCase, LowerCase, Reverse };
public static class FieldOperation {
    @FieldSelect(caption="Field")
    public String field;
    @DropList(caption="Operation")
    public Op operation;
    @Textbox(caption="Append")
    public String value;
}
@Table(caption="Edit operations")
public List<FieldOperation> fieldOperations;

This creates a UI element that lets the end user insert, remove, and reorder a list of FieldOperation elements. This is expressed as a three-column grid, where each column corresponds to a field in the FieldOperation class.

Your control can assume that the List<> is non-null and populated with non-null elements when configure() is called. However, you will probably need to skip over any "empty" rows, as it is very easy for users to accidentally enter these, as seen in the tool's example code.

CODE
for (FieldOperation op : fieldOperations) {
    if (op.field == null || op.field.isEmpty() || op.operation == null) {
        continue;
    }
    ...
}

Special consideration for table row class

Your table row class must have a default constructor! The class that represents the rows in your table control (the X in List<X>) cannot use the full complement of annotations, and there are some special cases.

The following annotations are not valid for fields of a table row class:

  • @DataView

  • @DynamicCaption

  • @GroupStart

  • @GroupEnd

  • @Menu

  • @NestedSettings

  • @OverridableSettings

  • @PushButton

  • @Schema

  • @SchemaEditor

  • @Tab

  • @Table

When using any of the following annotations:

  • @DynamicEnabled

  • @DynamicItems

  • @DynamicVisible

The referenced field always refers to a field in the outermost enclosing class (i.e. your tool class). The row items themselves cannot contain the dynamic values.

@DataView

This annotation is used to create "sample data" grids in the UI. For example, if your tool reads from a file or database, you might want to show a sample of the records that will be read from that data source. To use this control annotation, you need a field of type List<List<String>>.

CODE
@DataView(caption="Sample")
public List<List<String>> sample = new ArrayList<>;

You can populate this sample data anywhere in your code, but a good place is in response to a UI Action (see later topic), especially one resulting from a PushButton control.

Populate the control such that the first List<> element contains the field names for your sample data. This will be used to populate the row header in the display grid. The remaining List<> elements contain the sample data. We recommend that you keep the size of the data reasonable, (no more than 100 columns or rows) to avoid excessive memory use and UI sluggishness. See the DataViewSampleTool for sample code.

CODE
@PushButton(caption="Load sample")
@Action
public int button;
     
@DataView(caption="Sample")
public List<List<String>> sample = new ArrayList<>();
 
public void handleUIAction(String property, String action, UIActionState state) {
    if (property.equals("button")) {
        loadSample();
    }
}
private void loadSample() {
    sample = new ArrayList<>();
    List<String> columns = new ArrayList<>();
    sample.add(columns);
    int nbrCol = 10;
    for (int i = 0; i < nbrCol; i++) {
        columns.add("F" + i);
    }
    int nbrRow = 10;
    for (int rowIdx = 0; rowIdx < nbrRow; rowIdx++) {
        List<String> row = new ArrayList<>();
        sample.add(row);
        for (int colIdx = 0; colIdx < nbrCol; colIdx++) {
            row.add("V-" + rowIdx + "-" + colIdx);
        }
    }
}

The result looks like this.

Layout annotations

Data Management lets you control the layout of your user interface using the Label, GroupStart, GroupEnd, and Tab annotations.

@Label

Most control annotations include a caption element that you would normally use to label each control in your user interface. However, you might want to display more text than easily fits in a control's caption. In addition, @Label has some other useful aspects.

Think about using the @Label annotation when:

  • You want to display more text.

  • You want to change the label text dynamically.

Using the color, fontSize, and wrapping elements provided by the Label annotation, you have additional control over how your label is presented.

The following code snippet will result in a UI that looks like the following image.

CODE
@Label(color="#FF0000", fontSize=14)
public static final String label1="Enter the API ID associated with your account.";
@Textbox(caption="API ID")
public String accessKey;
@Label(color="#00FF00", fontSize=12)
public static final String label2="Enter the API passkey associated with your account.";
@Textbox(caption="Passkey")
public String secretKey;

@GroupStart, @GroupEnd

You can organize your controls into logical groupings by wrapping them in @GroupStart and @GroupEnd annotations. Data Management supports two group styles:

  • Collapsing, in which the group can be collapsed/expanded

  • Box, where the group is static and serves only to provide visual delineation

Group annotations can be useful when you need the contents of two or more groups to be visible at the same time. If you've got lots of controls, consider organizing them using Tab annotations.

Both of these annotations should be placed on a separate String field that contains the title text (while @GroupEnd never has any text, you have to put it on a String field anyway to give it something to be associated with).

For example, this code will place a collapsible group around the file browse controls (the indentation is subtle).

CODE
@GroupStart
public static final String groupStart1="Files";
@FileBrowse(caption = "Input file", filter = "Delimited files (*.csv)|*.csv")
public FilePath filePath;
@FileBrowse(caption = "Error output file", action = "savefile")
public FilePath errorFilePath;
@GroupEnd
public String groupEnd1;
@Textbox(caption = "Code page")
public String codePage;

@Tab

You can also organize related controls using a familiar tab metaphor with the @Tab annotation. In the following example we create two tabs, "Input" and "Output", and add a few controls to each.

CODE
@Tab
public static final String tab1="Input";
@FileBrowse(caption="Output file")
public FilePath inputFile;
@Spin(caption="Skip rows")
public int limitRows;
@Checkbox(caption="Strict schema checking")
public boolean validate;
     
@Tab
public static final String tab2="Output";
@FileBrowse(caption="Output file")
public FilePath outputFile;
@Checkbox(caption="Overwrite existing file")
public boolean overwrite;

Data Management generates the following control layout.

If you do not specify any @Tab annotations, a tab will be created for you titled "Configuration".

Unless @Tab is the first UI annotation in your class, it will create a new tab.

Use static and final on caption fields!

You'll notice in the above example that the String fields annotated with @Tab, @GroupStart, and @Label are marked with the Java static and final keywords. This tells Data Management that the text cannot change, and makes the value available to the UI generator. This in turn lets Data Management create a more efficient UI (since it will not have to always tell the client what is in the label). You should always use static final if the label text doesn't change. On the other hand, if your label text can change, you should omit these keywords.

Dynamically hiding and showing controls

@Trigger

Triggers make the visibility or enabledness of a control (or group of controls) dependent on the value of some other control. For example, suppose that your tool has a "logging" option expressed with a checkbox control; when that box is checked, you want a file-browse control to appear so that the user can select the log file, and a spin control to appear so that the user can set the desired log detail level. You can accomplish this using the @Trigger annotation placed on the checkbox control, as in the following example.

CODE
@Checkbox(caption="Enable logging")
@Trigger(targetControl="logging")
public boolean enableLogging;
@GroupStart
public static final String logging="Logging options"
@FileBrowse(caption="Log file")
public FilePath logFile;
@Spin(caption="Log level",minValue="0",maxValue="4")
public int logLevel;
@GroupEnd
public String groupEnd;

In the above example, we attach the Trigger annotation to a source control (the enableLogging checkbox) and reference a target control (the logging group). This binds the display of the logging group to the enableLogging checkbox, such that the contents of the Logging group are shown only when enableLogging is checked.

The targetControl refers to the fieldname of the target control, not its caption.

This trigger functionality enables you to build smarter, cleaner interfaces that only reveal properties when they become relevant.

The @Trigger annotation can be used with controls other than checkbox, by setting its other attributes:

  • compareToValue: what value of source control should be compared?

  • compareToValues: a list of source control values to compare.

  • comparison: the comparison operation (defaults to =).

  • caseSensitive: is the comparison case-sensitive? (defaults to false).

  • targetAttribute: the target attribute, one of visible or enabled. Defaults to visible.

  • targetTrueValue, targetFalseValue: the target attribute values. Defaults to true and false, respectively. Use these to invert the sense of the trigger (e.g. show when checkbox is NOT checked).

Refer to the JavaDoc for a complete description of this annotation's attributes.

@MultiTrigger

The @Trigger annotation can only specify a single trigger action on a control. While this is usually sufficient, there are times when you want one control to have many trigger effects, especially for dropdown lists where you often want to enable or show different controls or groups based on the selection. To handle those situations, use the @MultiTrigger annotation, which attaches multiple triggers to one control. With @MultiTrigger, you specify an array of @Trigger annotations. For example, the following creates a dropdown control with six choices, enabling the visibility of other controls based on the selection.

CODE
@MultiTrigger(triggers={
    @Trigger(compareToValue="one", targetControl="labelOneTwo"),
    @Trigger(compareToValue="two", targetControl="labelOneTwo"),
    @Trigger(compareToValues="three,four", targetControl="labelThreeFour"),
    @Trigger(compareToValues="five,six", targetControl="labelFiveSix"),
    @Trigger(compareToValue="one", targetControl="checkOne", targetAttribute="enabled"),
    @Trigger(compareToValue="two", targetControl="checkTwo", targetAttribute="enabled"),
    @Trigger(compareToValue="three", targetControl="checkThree", targetAttribute="enabled"),
    @Trigger(compareToValue="four", targetControl="checkFour", targetAttribute="enabled"),
    @Trigger(compareToValue="five", targetControl="checkFive", targetAttribute="enabled"),
    @Trigger(compareToValue="six", targetControl="checkSix", targetAttribute="enabled")
})

This effectively attaches a list of @Trigger annotations to a single control.

We can use either one @Trigger with the compareToValues attribute and a comma-separated list of values, or multiple @Trigger annotations, each with a single compareToValue.

See the PropertiesSampleTool example for a demonstration of this in action.

@DynamicEnabled and @DynamicVisible

Generally, using triggers is the correct way to hide and show other controls in a dependent fashion. However, there are times when you want your code to control the enabledness or visibility of a control or a group of controls. In these cases, you can use the @DynamicEnabled and @DynamicVisible annotations to influence control state via the values of other fields.

Example:

CODE
private boolean enableIt;
@DynamicEnabled(field="enableIt")
@Spin(minValue="1", maxValue="100")
public int value1;

The spin control will only be shown if your code sets the value of enableIt to true.

If your dynamic condition field is also a @Checkbox control, you should add the @Action annotation to make the UI respond immediately to changes in the condition as shown in the following snippet.

CODE
@Checkbox
@Action
public boolean enableIt;
@DynamicEnabled(field="enableIt")
@Spin(minValue="1", maxValue="100")
public int value1;

@DynamicEnabled and @DynamicVisible: using getter methods

The following is available starting with RPDM version 9.2.

Starting with RPDM version 9.2, you can use getter methods in addition to fields for @DynamicVisible and @DynamicEnabled.  This follows the typical java naming convention for getters, so if the field name is xxx, the method is getXxx().  Using getter methods is more flexible when you have a class hierarchy of tools, because derived tools can override the getter methods.

CODE
@DynamicEnabled(field="enableIt")
@Spin(minValue="1", maxValue="100")
public int value1;
 
public boolean getEnableIt() { custom logic }

Dynamic content

You have already seen above how the @Tab, @GroupStart and @Label annotations derive their text from the value of a following String field. Similarly, you can populate the contents of some controls by referencing the values of fields in your class. The content of these controls is dynamic, meaning that you can change the field value, and the UI will reflect that change. Changes made to dynamic fields will be reflected in the UI the next time it loads data “from the server,” specifically:

  • When the properties for the tool are loaded (the user selects the tool)

  • In response to a UI action event (the user operates a UI control that was annotated with @Action)

@DynamicCaption

This annotation specifies that the caption for a control should be derived from a String field. You can see in the example below that the caption of the text control will be obtained from the caption field.

CODE
@DynamicCaption(field="caption")
@Textbox
public String name;
private String caption="Name";

You should make the caption field itself private, to prevent Data Management from turning it into a UI control in its own right.

@DynamicItems

This annotation lets you populate the contents of a dropdown list (in a combo box control) from a list of strings stored in a field in your class. For example, it can be used to populate a dropdown control with a list of database table names. This is especially useful when combined with a @PushButton control that triggers an action in your code (see Dynamically hiding and showing controls). For example, the following code will populate the contents of a dropdown control from a list of Strings stored in a field named tableList.

CODE
@DropList(caption="Table")
@DynamicItems(field="tableList")
public String table;
private List<String> tableList;
 
private void fillTableList() {
   tableList = new ArrayList<>();
   for (Table t : database.fetchTables()) {
     tableList.add(t.getName());
   }
}

Separating drop-list elements display and value

This feature is available in 8.1.0 and later.

Sometimes, the elements you want to put in a drop-down control have both a "human readable" display part and an internal "unique identifier" part. To use this technique, nominate a List<net.redpoint.dataflow.ui.DropListItem> as the field() of the @DynamicItems annotation. The user will then see and choose the "display" elements, but the target field will contain the chosen "value".

Example:

CODE
import net.redpoint.dataflow.ui.DropListItem;
@DropList(caption="Table")
@DynamicItems(field="tableItems")
public String tableId;
private List<DropListItem> tableItems;
 
private void fillTableList() {
   tableItems = new ArrayList<>();
   for (Table t : database.fetchTables()) {
     tableItems.add(new DropListItem(t.getUniqueId(), t.getName()));
   }
}

The tableId field will contain the selected element from tableIdsnot from tableNames.

Invoking methods in your tool's code

@Action

The @Action annotation wires up a UI control to a method in your UI. This allows your code to do things like:

  • Connect to a database

  • Analyze a file format

  • Display a sample of the input data

  • Fire the rocket thruster

While any control can be annotated with @Action, we recommend that you limit its use to one of several control types:

  • checkbox

  • dropdown list

  • pushbutton

The reason for this limitation is that other control types can fire the action quite frequently, and we've found in practice that this leads to laggy UIs with odd behavior. In future versions of Data Management we plan to add governors to the action rates to avoid these issues.

To enable your class to handle actions, it must implement the HasDynamicUI interface, which contains two methods:

public void handleUIAction(String property, String action, UIActionState state);

This method is called when a control annotated with @Action is operated. The parameters to this method are:

  • property: the name of the control that was operated (the field name to which the annotation was bound).

  • action: the type of operation performed, so that for example you can tell the difference between typing and selecting in a combo box.

  • state: an object that you can update with a message and percentage completion during long-running operations, and indicates when the user has cancelled the operation.

public void refreshUI();
This method is called when Data Management wants to update the dynamic content of controls, and gives you a chance to fill in dynamic-content items based on the current state. Normally you can just implement this with an empty method.

For example, the @DataView example code shown above calls the loadSample() method. It is wired to the pushbutton control as follows.

CODE
@PushButton(caption="Load sample")
@Action
public int button;
     
public void handleUIAction(String property, String action, UIActionState state) {
    if (property.equals("button")) {
        loadSample();
    }
}

The @Action annotation has three modes:

  • wait: the default mode, this causes the UI to pause until the action is complete. This is useful for short actions or places where you don't mind making the user wait.

  • modal: this will cause the UI to pop up a dialog that can display a message and completion status, and which also has a cancel button. Use this for long-running actions with progress feedback.

  • modeless: this will cause the UI to run the action silently in the background. When the action is complete, any fields that you have changed in your tool will be updated in the UI as well.

@Pushbutton

The @Pushbutton annotation exists solely to make a pushbutton control appear in the UI. It serves no purpose other than calling your handleUIAction() method when it is pressed. Use it anywhere you want your tool's code to "do something now."

You've seen it used in the DataViewSampleTool already, but you can also see it in the AnalyzeSchemaSample tool. In this sample, the pushbutton is wired up to fill in a Schema field with fake data when it is pressed.

CODE
@PushButton(caption="Analyze")
@Action
public int button;
     
@SchemaEditor(caption="Schema")
public Schema schema;
 
 
public void handleUIAction(String property, String action, UIActionState state) throws ErrorException {
    if (property.equals("button")) {
        analyze();
    }
}
 
 
private void analyze() throws ErrorException {
    List<FieldSpec> fsl= new ArrayList<>();
    for (int i = 0; i < 10; i++) {
        FieldSpec fs = new FieldSpec("F" + i, DataType.Text);
        fsl.add(fs);
    }
    try {
        schema = factory.getTheSchemaFactory().makeSchema(fsl);
    } catch (DuplicateNameException ex) {
        throw new ErrorException("Invalid schema", ex);
    }
}

After the button is clicked, the result looks like this.

Of course, this is fake data, but the idea is that your tool could reach into the input file (or database connection, etc.), analyze the schema of its input data, and display that schema to the user.

@Menu

The @Menu annotation makes a menu control appear in the UI. It can annotate any field—the type is ignored and the field value is not used. Its only purpose is calling your handleUIAction() method when an item is selected. Use it anywhere you want your tool's code to "do something now." A menu is more compact than a pushbutton for multiple action choices.

When activated, your handleUIAction() method will be called with property equal to the menu item (not its text, if also specified—see below).

Menu items

Set the items property of the Menu annotation to specify your menu items. Menu items are separated by commas, for example: items="item1,item2,item3".

Make the menus more readable by adding text in parentheses following each menu item: items="item1(Menu Item1),item2(Menu Item2)".

The menu will display the text (e.g. Menu Item 1) but the action will receive the label (e.g. item1). This helps your UI separate the human-readable strings from the programatically-processed strings.

Add submenus by enclosing the sub-list in square brackets following the sub-menu item items="item1,Sub Items[subitem1,subitem2(Sub Item2)]".

There is no way to add any of these characters to a menu item: ( ) [ ] ,

Example

You can also see this annotation used in the MenuControlSampleTool example. In this sample, the menu is wired up to set a label when selected.

CODE
@Label
public static final String menu1Label = "Menu with items only";
@Menu(caption="Simple",items="menu1_item1,menu1_item2,menu1_item3")
public int menu1;
 
@Label
public static final String menu2Label = "Menu with icon";
@Menu(iconResource="/net/redpoint/dataflow/samples/Menu_16.png",items="menu2_item1,menu2_item2,menu2_item3")
public int menu2;
 
@Label
public static final String menu3Label = "Menu with item text";
@Menu(caption="Item text",items="menu3_item1(Menu Item 1),menu3_item2(Menu Item 2),menu3_item3(Menu Item 3)")
public int menu3;
 
@Label
public static final String menu4Label = "Menu with submenus";
@Menu(caption="Sub Menus",items="menu4_item1,submenu(Sub Menu Items)[menu4_sub1,menu4_sub2(SubItem2)],menu4_item3")
public int subMenus;
@Label
public String lastAction = "This label will be filled with menu selection";
 
@Override
public void handleUIAction(String property, String action, UIActionState state) {
    lastAction="Menu selected:" + property;
}

Your tool can perform more meaningful actions in handleUIAction, such as analyzing a file or connecting to a database.

Shared settings

During development of your SDK tools, you may find it convenient to store some settings globally, so that the user of your tools doesn't need to enter the same settings for every instance of the tool. For example, if your tool connects to a server using an URL, and the normal use case is for all instances of the tool to connect to the same server, you could make a shared setting for the server URL.

Shared settings are stored in the repository in the object path /Settings/Tools. All shared settings are in the same place. Your tool's settings will appear on a tab of their own to distinguish them from other settings.

See the SharedSettingsTool in the dataflow_examples project for a detailed example.

Defining shared settings

To define shared settings, you must do two things. Once you have done this and dropped your tool's jar file into the java_plugins folder, RPDM will automatically add a tab to /Settings/Tools in the repository and create a user interface for editing the properties.

Implement net.redpoint.dataflow.transform.Settings

In the SDK there is a simple abstract class named Settings.

CODE
public abstract class Settings {
    /**
     * Name of the settings; this will be turned into the tab label in /Settings/Tools
     * @return The name of the setting category
     */
    public abstract String getName();
}

You should implement a subclass of Settings, returning your settings category, and defining and annotating your settings fields, for example.

CODE
public class MySettings extends Settings {
    @Override
    public String getName() { return "GeoServer"; }
     
    @Textbox(caption="GEO Server URL")
    public String geoServerURL;
    @Textbox
    public String username;
    @Textbox(caption="Set password", password=true)
    public String password;
}

Add a resource to META-INF.services

In your tool's project, add a resource named META-INF.services/net.redpoint.dataflow.transform.Settings, as shown in Netbeans.

Add an entry for your settings class to this resource file.

CODE
#Definition of services provided by my tools package
com.mycompany.myproject.MySettings

Using shared settings

Given a built jar with your shared settings object copied into install/java_plugins, the RPDM user will be able to edit the shared settings in the repository. But your class must also fetch the settings at run-time, and use them. This is done by calling getSettings() in the ToolContext.

CODE
MySettings settings = context.getSettings(MySettings.class);

This call does several things for you:

  • Reads /Settings/Tools from the repository and parses the section for your settings.

  • Creates an object of the correct type.

  • Uses reflection to populate the settings object's fields in the same way the tool properties are set from the tool configuration.

Given your settings object, just access its fields directly.

Use @OverridableSettings to automatically incorporate shared settings

This feature is available in 8.0.1 and later.

The @OverridableSettings annotation, when used in conjunction with Shared settings, lets you manage complex properties and a generate a UI that automatically incorporates the system-wide settings while allowing the end user to override those settings for each tool. For example, let's start with the shared settings used in the samples pack ExampleSettings.java.

CODE
public class ExampleSettings extends Settings {
    @Override
    public String getName() {
        return "SDKExample";
    }
    @Spin(minValue="1", maxValue="100")
    public Long number;
    public String login;
    @Textbox(password=true)
    public String password;
}

When you add a Settings subclass to your META-INF.services/net.redpoint.dataflow.transform.Settings resource file, it will cause a tab to be automatically created under /Settings/Tools (in this case, the SDKExample tab). If you have installed the examples jar, bring up this tab in Data Management and commit these settings.

Now that you have a defined a Settings object and its properties are folded into the /Settings/Tools UI, you are ready to use it in a tool. To do so, define a field of your Settings type and annotate it with @OverridableSettings. For example, consider the samples pack SharedSettingsTool.java.

CODE
@GroupStart(style="box")
public String overrideGroup = "Overridable settings";
@OverridableSettings
public ExampleSettings settings;

If you have the samples jar installed, create a project, drag in the SharedSettingsTool, and you'll see these properties.

How the default properties that were set in /Settings/Tools->SDKExample in the repository automatically populate the properties of this tool when Override is unchecked. This serves as a visual cue to the user that these default settings will be in use. If you attach a Data Viewer tool and run the project, these defaults are shown in the data output.

When the user checks Override, she can enter values that are specific to this tool instance.

And when run, the project will show the overrides as output data.

Unchecking Override again restores the tool to its default settings.

Don't call ToolContext.getSettings(). Use @OverridableSettings and the correct settings will be automatically chosen and injected into your settings field.

You cannot use the following annotations in a Settings class, because the referenced fields will never be populated when the UI is shown under /Settings/Tools:

  • @DynamicItems

  • @DynamicCaption

  • @DynamicEnabled

  • @DynamicVisible

Use @NestedSettings to organize properties into another class

This feature is available in Redpoint Data Management 7.4.2 and later.

It is often convenient (especially when creating a set of tools with shared behavior) to declare a field that is a class instead of a primitive type. This class, in turn, contains annotated fields. If you annotate such a field with @NestedSettings, the SDK will automatically inject properties for the field's class type, for example.

CODE
public class GeoStuff {  
    @Textbox(caption="GEO Server URL")
    public String geoServerURL;
    @Textbox
    public String username;
    @Textbox(caption="Set password", password=true)
    public String password;
}
 
public class MyTool extends Tool implements InputTool {
   ...
   @GroupStart
   public static final String l1 = "Geo Stuff";
   @NestedSettings
   public GeoStuff geoStuff;
}

This will result in the following UI.

The SDK will automatically get and set the "geoStuff" field.

See the SharedSettingsTool example in the dataflow.samples project for a detailed sample.

Generated names of elements inside @NestedSettings

When you annotate a field with @NestedSettings, Data Management will generate a name for the referenced field of the form: outerFieldName + "." + innerFieldName.

This is the name that is used in all of the following annotations.

Annotation

Property

Trigger

targetControl

DynamicVisible

field

DynamicCaption

field

DynamicEnabled

field

DynamicItems

field

This implies that @Trigger annotations used outside of the nested field must reference the decorated name. However, @Trigger inside the nested class should still use the undecorated name.

For example, consider the following code.

CODE
class MySettings extends Settings {
   // This trigger on the inside uses the undecorated name, as usual
   @Trigger(targetControl="inner1")
   boolean check1;
   public String inner1;
   public String innter2;
}
class MyTool extends Tool ... {
   // This trigger from the outside must use the decorated name
   @Trigger(targetControl="nested.inner2")
   boolean check2;
   @NestedSettings
   public MySettings nested;
}

In addition, nested settings can refer to elements in the containing Tool class in @Dynamic* annotations by prefixing the name of the field with "*.".

Example:

CODE
class MySettings extends Settings {
   // Refers to elements in the containing class
   @DynamicItems(field="*.items")
   @DropList
   String dropList;
   // Refers to field in the containing class
   @DynamicEnabled(field="*.enabled")
   @Textbox
   public String editMe;
}
class MyTool extends Tool ... {
   @NestedSettings
   public MySettings nested;
   // Supplies items for nested.dropList;
   private List<String> items;
   // Controls enabled status of nested.editMe
   @Checkbox
   public boolean enabled;
}

Without the "*.", such field references refer to fields inside the nested class, not the containing class. In practice the containing class is always a Tool. While you can put @NestedSettings inside other @NestedSettings, there is no way to refer to fields in the intermediate nesting levels.

Use @SuperSettings to inherit settings and behavior from a superclass

This feature is available in 7.4.2 and later.

We've seen above that @NestedSettings is useful for encapsulating sets of settings to another class, which also makes it easier to use and override "shared settings" from the repository. However, you may also find it useful to use inheritance in your SDK tools, allowing them to share both settings and processing logic. For example, suppose that you want to create a set of tools for connecting to a new database. You might create a base class tool that has a set of connection properties and handles such things as testing the connection and populating the database and table list, but implement the specific operations (load, query, execute) in subclasses.

If you do not add the base class to your META-INF.services/net.redpoint.dataflow.transform.Tool resource, the base class will not show on the palette or be usable as a tool in its own right. This is usually what you want—the derived classes are on the palette but the base class is not.

See the SuperSettingsToolBase and SuperSettingsToolDerived tools in the dataflow.samples project for an example of this pattern.

How do user interface properties get set into your class fields?

Data Management will automatically interpret the user-settable properties and set your object's public or annotated data members before calling your tool's configure() method. So when configure() is called, you can assume that the fields corresponding to UI elements have already been set to reflect the user settings.

Your configure method is also a logical place to validate your properties.

CODE
@Override
public Schema configure(DataItem config) throws ErrorException {
  if (filePath.isEmpty()) {
    throw new ErrorException("File path not specified");
  }
  return schema;
}

The configure method is called when:

  • A project containing your tool is opened

  • A new instance of your tool is dropped on the canvas

  • Changes to the Property panel are committed to your tool

  • Changes to "upstream" connected tools are made that result in different Schemas being presented to your tool

  • The user requests a refresh of file or database connections

For additional detail on the configure method, see Understanding the Tool lifecycle.

Discontinued use patterns and behavior changes

The following usage patterns and interface elements have been deprecated. New code development should discontinue their use.

Manual shared settings pattern using ToolContext.getSettings()

Long ago, the SDK supported a use pattern of "manually overriding shared settings". This was done first by defining properties like this.

CODE
@Checkbox(caption="Override shared settings")
@Trigger(targetControl="overrideGroup")
boolean override;
@GroupStart(style="box")   
public String overrideGroup = "Overridden settings";
@NestedSettings
public SDKExampleSettings overrideSettings;

Then processing those properties like:

This is no longer supported. ToolContext.getSettings() will not return the shared settings. You will get a NullPointerException.

Instead, use @OverrideableSettings.

Stacking @Label with other annotations

Prior to version 7.4, the only way to create a label control was to put the @Label annotation on something that was already going to be a control (like a String with @Textbox annotation). This would cause a label to appear above the intended control. For example, the following was supported.

CODE
@Textbox(caption="API ID")
@Label(caption="Enter the API ID associated with your account.", color="#666666", fontSize=10)
public String accessKey;

However, this created problems (for example, you could not control label visibility with @Trigger). We now recommend that you use @Label to annotate a String that contains the label text, as in the following.

CODE
@Label(color="#666666", fontSize=10)
public static final String labelText="Enter the API ID associated with your account.";
 
@Textbox(caption="API ID")
public String accessKey;

To reinforce this change, you will notice that the caption attribute of @Label has been deprecated. The caption should always be contained in the annotated String field instead.

Unless your label text can change dynamically, you should also add the static final keywords to your String field.

Stacking @GroupStart, @GroupEnd, and @Tab with other annotations

Prior to version 7.4, the only way to start a group or tab was to put the @GroupStart, @GroupEnd, or @Tab annotation on something that was already going to be a control (like a String with @Textbox annotation). This would cause a group to start above the targeted control. For example, the following was supported.

CODE
@Tab(caption="Input")
@FileBrowse(caption="Output file")
public FilePath inputFile;
@GroupStart(caption="Options")
@Spin(caption="Skip rows")
public int limitRows;

However, this created problems (for example, you could not control group visibility with @Trigger, and there were intractable ambiguities). We now recommend that you use @Tab or @GroupStart to annotate a String that contains the title text of the tab or group, as in the following.

CODE
@Tab
public static final String tabTitle="Input";
FileBrowse(caption="Output file")
public FilePath inputFile;
@GroupStart
public static final String groupTitle="Options";
@Spin(caption="Skip rows")
public int limitRows;

To reinforce this change, you will notice that the caption attributes of @Tab and @GroupStart have been deprecated. The caption should always be contained in the annotated String field instead.

Unless your caption text can change dynamically, you should also add the static final keywords to your String field.

Tool.getInterfaceXml()

Prior to RPDM version 7.4, you could override getInterfaceXml() to generate custom user-interface XML. As of version 7.4 this is no longer supported, and the Tool.getInterfaceXml() method is final.

Generated names of elements inside @NestedSettings and @OverridableSettings

As of version 8.0, fields inside the type referenced by @NestedSettiings and @OverridableSettings will be assigned a name (used for trigger and dynamic content purposes) that contains the enclosing field name as a prefix. See the section on @NestedSettings for details.

Erroneous annotations

In Data Management version 7.4, incompatible annotations were largely ignored or sometimes processed incorrectly. For example, you could annotate the same field with @Spin and @Textbox. Either annotation might be applied. In Data Management version 8.0, extra incompatible annotations will result in an "error" label being generated in the UI to inform you that something is wrong. Error labels are large and red to make them hard to miss. For example, all of the following will result in error labels.

CODE
// @Label attached to a date field, but this can only be done with String
// You need to specify the caption of the @Label to use it this way
@Label
public Date date;
 
// @Label attached to something that is marked as a control.
// You need to specify the caption of the @Label to use it this way
@Label
@Checkbox
public boolean checkme;
 
// Multiple control annotations cannot be stacked on the same field
@Spin
@Textbox
public double value;
 
// enum fields cannot be anything except @Droplist
enum Foo { FOO1, FOO2, FOO3 };
@Textbox
public Foo foo;

In addition, using any of the disallowed annotations from special table considerations will result in an "error label" for that column in the table.

Non-public fields

Prior to version 8.0, non-public fields with any SDK annotation would cause them to be treated as public. Now, only annotations that designate the field as a control will cause non-public fields to produce controls in the UI. For example, the following will no longer produce a checkbox.

CODE
@Label(caption="Hi there")
boolean ignoreMe;

Use of defaultValue in annotations

Prior to version 8.3.1, most control annotations had a defaultValue property, the intent of which was to set the default value of a field in the UI, as in.

CODE
@Textbox(caption="Queue name", defaultValue="default")
String queue;

However, this never really worked correctly – the default value would show in the UI, but it would not be used in the SDK tool unless you opened the UI and committed it.

Because of this all defaultValue have been deprecated. As of version 8.3.1, the supported way of setting default values is to simply set the field value, either in the constructor or inline in the class.

CODE
@Textbox(caption="Queue name")
String queue = "default";

JavaScript errors detected

Please note, these errors can depend on your browser setup.

If this problem persists, please contact our support.