This SDK is a work in process. Please open an issue as you encounter them. Not all features are implemented at this time and bugs may be common for the time being. This SDK is offered 'as-is' and is not a supported product.
You can find example implementations of this SDK here: https://github.com/jmccuen/ThingworxPlotlyExamplePlots
These charts are built on top of the Plotly Javascript Library, detailed documentation for which can be found here: https://plot.ly/javascript/ bui
- Download the SDK zip and install normally, as a Thingworx Extension: https://github.com/jmccuen/ThingworxPlotlyChartingSDK/blob/master/zip/
- Install any extension charts which utilize this SDK, such as the example chart's extension zip: https://github.com/jmccuen/ThingworxPlotlyExamplePlots/blob/master/build/distributions/PlotlyPlots.zip
- Clear cache and refresh Composer
- Example Charts should appear as Timeseries Plot, Label Plot, Pie Plot, and Surface Plot in the Mashup builder as normal
The purpose of this SDK is to ease the process of building charting widgets in PTC's Thingworx platform. It is built off of the standard Thingworx Widget API. For more details on developing Thingworx Widgets, please see the documentation here: https://www.ptc.com/support/-/media/FFF75A096E3245C8BA1E42E1C47C04CD.pdf?sc_lang=en
Charts share many things in common:
- Chart Area Styling
- Chart Title
- Chart Axes
- Styling
- Naming
- Position
- Multiple Axes
- Legends
- Styling
- Position
- Data fields
- Styling
- Tooltips
- Events
- Selection
- Double-click
- Data bindings
- Services
- Zoom
- Pan
- Resize
Many of these items are the same across charts of all types, or are shared by several subtypes. When creating widgets in Thingworx, every new widget has to solve these problems again for themsevles. This causes a lot of boilerplate code, repetition, large file size of the combined widget javascript files, and makes many things -- such as fixing bugs, improving performance, maintaining old charts, creating new charts, and implementing new features across charts -- very difficult.
This SDK simplifies all of these things by putting common functionality in one place. Contained in this package is a widget extension with a common charting library -- plotly.js, a popular open-source MIT licensed library -- and two custom Thingworx wrapper libraries for the IDE and Runtime, respectively.
First, install the current build of the extension, kit.zip, on your instance of Thingworx. This will install Plotly and the two Thingworx libraries.
Begin widget development as normal (see documentation above)
At the beginning of your chart's ide.js, add the following:
let chart = new TWIDEChart(this,CHART_NAME, CHART_CSS_CLASS, MAX_SERIES,TYPE,MAX_AXES,MULTIPLE_DATASOURCES);
Where 'this' is your widget instance and the rest are the parameters as named. Currently supported types are 'pie', '2d', and '3d' -- these may be extended as needs arise. This call will create a new widget for you based on the type of widget you are creating.
The charting SDK works similarly to an abstract class, thus when you implement a widget API callback, you can also choose to implement the chart's function. For example, your widget.afterRender() callback my look like the following:
this.afterRender = function() {
return chart.afterRender();
};
If you want to add additional functionality to your chart, you can add it to your callback function either in addition to or in place of the chart function.
this.updateProperty = function (updatePropertyInfo) {
chart.updateProperty(updatePropertyInfo);
if (updatePropertyInfo.TargetProperty === 'Data') {
let data = getData(updatePropertyInfo,false);
chart.draw(data);
};
for (let i=1;i <= Number(properties['NumberOfSeries']);i++) {
if (updatePropertyInfo.TargetProperty === 'DataSource'+i) {
if (updatePropertyInfo.ActualDataRows.length > 0) {
let data = getData(updatePropertyInfo, true);
chart.draw(data);
};
};
};
};
Most of the standard widget properties are already available via the implementation described above:
this.widgetProperties = function () {
let properties = chart.widgetProperties();
return properties;
};
As with other callbacks, you can extend this or replace it with the properties you need. You can also modify any of the properties by modifying the object -- whatever is returned by this function is what will be added into your chart in the IDE.
Notes:
- Additional properties can be added directly to properties.properties object.
- Thingworx orders properties in the mashup builder based on the order in which they are added to the JSON object. This means extended properties will come at the bottom of the properties object. We are currently working on an approach to solve this problem, so that properties can be inserted in logical places.
- Properties must be implemented in this callback, you cannot dynamically add properties to the widget in a later call. You must show/hide those properties.
At the beginning of your widget runtime.js, add the following:
let chart = new TWRuntimeChart(this, CHART_CSS_CLASS);
This will create an instance of the runtime chart with your widget.
As with the ide.js, the chart SDK implements most of the standard widget callbacks which can be extended or replace:
this.renderHtml = function () {
return chart.renderHtml();
};
First, make sure to tell the chart a property has been updated:
chart.updateProperty(updatePropertyInfo);
This ensures that common bindings -- such as Chart Title or Axis Title -- are updated automatically.
Then, handle the update as it relates specifically to your chart. Update property is where most of the work for your widget will happen. Here, you will need to take in an InfoTable and produce the correct format for Plotly to render. Please see https://plot.ly/javascript/ for details. When you have created your data array, call chart.draw(data).
Example data format:
var trace1 = {
x: [1, 2, 3, 4],
y: [10, 15, 13, 17],
mode: 'markers',
type: 'scatter'
};
var trace2 = {
x: [2, 3, 4, 5],
y: [16, 5, 11, 9],
mode: 'lines',
type: 'scatter'
};
var trace3 = {
x: [1, 2, 3, 4],
y: [12, 9, 15, 12],
mode: 'lines+markers',
type: 'scatter'
};
var data = [trace1, trace2, trace3];
chart.draw(data);
You will likely need to transform your InfoTable into a format that is suitable for your chart:
this.updateProperty = function (updatePropertyInfo) {
chart.updateProperty(updatePropertyInfo);
if (updatePropertyInfo.TargetProperty === 'Data') {
let rows = updatePropertyInfo.ActualDataRows;
let z = [];
for (let i = 0; i<rows.length;i++) {
let row = rows[i];
let x = properties['XAxisField'];
let value = properties['ZDataField1'];
if (!z[row[x]]) {
z[row[x]] = [];
}
z[row[x]].push(row[value]);
}
let trace = new Object();
trace.series = 1;
trace.z = z;
trace.type = 'surface';
chart.draw([trace]);
}
There are also helper functions exposed to make translating your chart into the correct data format easier.
chart.getXY(it);
getXY will take in an infotable and make the data to a values object that is returned. The values object will have a single X axis array and an array of Y Arrays, one for each trace, which are correctly mapped from the YDataField properties.
chart.getDynamicXY(it);
GetDynamicXY will take an InfoTable and, as long as an X Axis is mapped, will map the remaining items that are NUMBERS or INTEGERS to corresponding traces, even if no selection has been made for the series. The NumberOfSeries properties are still enforced.
chart.layout
chart.chartData
chart.chartInfo
chart.layout is the layout object that the charting SDK uses when rendering Plotly. You have full access to this and can update it at any time. You can also call Plotly directly on the chart div, using functions such as relayout, as necessary.
chart.chartData is the actual dataset that Plotly is rendering. You can make modifications directly to this item and update any changes with Plotly.react
chart.chartInfo is an object you can use to store any additional information you would like to about your chart for future reference. This object is not used by the charting SDK.