Creation

Operations associated with the creation of a DataTable object depend on specific implementation of DataTable interface.

SimpleDataTable

To create an instance of SimpleDataTable it's necessary to define its format by creating an instance of TableFormat class:

TableFormat tf = new TableFormat(); // For multi-row tables

TableFormat tf = new TableFormat(1, 1); // For single-row tables

In the majority of cases TableFormat instances should be declared static, configured in static blocks and re-used between multiple tables, variables, functions, events, etc. This ensures much lower memory consumption comparing to non-static table formats.

Adding Fields

After that, it's necessary to add field descriptors to a table format. Each field is represented by a FieldFormat instance. There are many ways to create a field format descriptor:

FieldFormat ff = FieldFormat.create("fieldName", FieldFormat.STRING_FIELD); // The simplest method to create a string field
ff.setDescription("Field Description");
ff.setDefault("Default Value");
ff.setNullable(true);

FieldFormat ff = FieldFormat.create("fieldName", FieldFormat.STRING_FIELD, "Field Description", "Default Value", true); // Same as above, but specifying all options in constructor

FieldFormat ff = FieldFormat.create("<fieldName><S><D=Field Description><A=Default value><F=N>"); // Same as above, but using [field format encoding] link below.

// After field format is created, we can fine-tune it
ff.addSelectionValue("v1", "First option");
ff.addSelectionValue("v2", "Second option");
ff.addValidator(new LimitsValidator(10, 100));
ff.setKeyField(true);

// Now we can add our field descriptor to field format:
TableFormat tf = new TableFormat();
tf.addField(ff);

Field Format Encoding

It's also possible to avoid explicit creation of FieldFormat and pass field options to TableFormat.addField():

TableFormat tf = new TableFormat();

tf.addField("fieldName", FieldFormat.STRING_FIELD, "Field Description", "Default Value", true);

Setting Format Options

When all fields were added, we can fine-tune out table format and add some bindings and validators to it:

tf.setReorderable(true);

tf.addBinding("field2", "{field1} * 2");

tf.addTableValidator(new TableKeyFieldsValidator());

Creating Table

When our table format is ready, we can base a table on it:

TableFormat tf = new TableFormat(1, 1, FieldFormat.create("fieldName", FieldFormat.STRING_FIELD));

DataTable table = new SimpleDataTable(tf);

Once TableFormat instance has been used to create a table, it becomes immutable and cannot be modified. To modify format of an existing table:

  • Clone format of the table: TableFormat cloned = table.getFormat().clone();
  • The cloned format will be mutable. Make necessary modifications to it.
  • Create a new table using cloned format: DataTable newTable = new SimpleDataTable(cloned);
  • Copy data from old table to a new one using a static helper method of DataTableReplication class: DataTableReplication.copy(table, newTable);

Adding Records

There are several ways to create a record.

First method - setting field values explicitly:

DataRecord rec = table.addRecord();
rec.setValue("firstField", "Value of first field"); // First field is a string-type field
rec.setValue("secondField", true); // First field is a boolean-type field
rec.setValue("thirdField", 12345); // First field is an integer-type field

Second method - passing field values to addRecord():

DataRecord rec = table.addRecord("Value of first field", true, 12345);

Third method - adding values one by one:

DataRecord rec = table.addRecord();
rec.addString("Value of first field");
rec.addBoolean(true);
rec.addInt(12345);

// Explicit type specification is not necessary
DataRecord rec = table.addRecord();
rec.addValue("Value of first field");
rec.addValue(true);
rec.addValue(12345);

Automatic Type Conversion

Sometimes a value that you may have might not be directly suitable for a table cell. In this case, use setValueSmart() method of a DataRecord. It will do its best to convert value to a proper type.

Internally, the code is using static methods of Util class that can be also used directly to enable conversion of raw objects to booleans, numbers of dates.

ResultSetDataTable

To create an instance of ResultSetDataTable, an instance of a class that implements java.sql.ResultSet is required. The example below demonstrates how to create a ResultSetDataTable using a MySQL JDBC connection:

DriverManager.registerDriver(((Driver) Class.forName("com.mysql.jdbc.Driver").newInstance())); // it is supposed that the required JDBC driver is already on the classpath
Connection con = DriverManager.getConnection("jdbc:mysql://localhost/databasename?user=root&password=123&serverTimezone=Europe/Moscow");
Statement stmt = con.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_UPDATABLE);

String query = "SELECT * FROM TABLENAME;"; // here it is supposed that there is a table called "TABLENAME" in the database called "databasename"
ResultSet resultSet = stmt.executeQuery(query);

DataTable table = new ResultSetDataTable(resultSet);
// Do not forget to close the instances of ResultSet, Statement, and Connection when they are no longer required

Note that the format of the constructed data table is created automatically, based on the schema of the relation represented by the result set.

Adding Records

To add records to a ResultSetDataTable, the result set that the table is based upon must be both scrollable (meaning that it must not be of type ResultSet.TYPE_FORWARD_ONLY) and updatable (ResultSet.CONCUR_UPDATABLE).

Records can be added in the same manner as described above for SimpleDataTable with one exception: the addRecord() method (with no arguments) is not supported. Method addRecord(DataRecord) should be used instead:

DataRecord rec = new DataRecord(table.getFormat);
rec.addString("Value of first field");
rec.addBoolean(true);
rec.addInt(12345);
table.addRecord(rec);

FilteringDataTable

As was mentioned in the previous section, a FilteringDataTable instance does not hold data. To create an instance of FilteringDataTable, at least two object are required: a data table that will serve as a source of data and a filter expression that will define the filtering condition(s). The filter expression can be either of type Expression or of type String. The latter will be automatically converted into an Expression instance (provided that the string is a valid Expression that is evaluated to a boolean).

Note that the DataTable instance that has been passed to a FilteringDataTable constructor becomes immutable.

The example below demonstrates the creation of a FilteringDataTable:

// Let's suppose we have a SimpleDataTable called "source" that has an Integer field called "intfield". This data table will serve as data source for our FilteringDataTable instance

String filterExpression = "{intfield} == 111 || {intfield} == 333";
DataTable filteringTable = new FilteringDataTable(source, filterExpression);

// The created FilteringDataTable will have only the records from source that satisfy the conditions defined in filterExpression (i.e. only records with the "intfield" value equal to 111 or 333)

FilteringDataTable does not support mutating operations so it is impossible to add or set records.

ProxyDataTable

An instance of ProxyDataTable is created automatically on the Client's side when the Client attempts to access an instance of ResultSetDataTable or FilteringDataTable on the Server.

ProxyDataTable does not support mutating operations so it is impossible to add or set records.

Was this page helpful?