Skip to content

This is an SDK based on the Plotly Javascript library to create charting widgets in Thingworx

License

Notifications You must be signed in to change notification settings

jmccuen/ThingworxPlotlyChartingSDK

Repository files navigation

Plotly Charting SDK

Disclaimer

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.

Examples

You can find example implementations of this SDK here: https://github.com/jmccuen/ThingworxPlotlyExamplePlots

Plotly

These charts are built on top of the Plotly Javascript Library, detailed documentation for which can be found here: https://plot.ly/javascript/ bui

Install instructions

Purpose

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.

How to use

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)

IDE

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);
        		};
        	};
		 }; 
	};

widgetProperties

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.

Runtime

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();
	};

updateProperty

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]);
			
		}

Helper functions and properties

There are also helper functions exposed to make translating your chart into the correct data format easier.

Runtime

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.

About

This is an SDK based on the Plotly Javascript library to create charting widgets in Thingworx

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published