Change Log
- 02/08/2015 - Blog posted
Planned Enhancements
- Up for discussion!
- Iterate via columns?
Description
Use Case:
Design Studio, being a multi-dimensional client, likes to decompose data into 'tuples' which basically defines each measure as a cell value that is represented by indexed tuples that map back to a dimension member. There's not really anything "wrong" with this, except that at least in my opinion, this is not the easiest way for me to work with the data in a 2D tabular form. There have been many SCN threads where people have struggled with basically looping through (aka iterating over) the rows in order to apply some script logic. Even in my other extensions, the first thing I do is "flatten" the dataset into a 2-dimensional (or table-like) form to work with. I start thinking about why couldn't I just open this convenience up to BIAL scripting? This is what I have written this component called 'Data Iterator'.
Welcome to the Data Iterator
The Data Iterator is a Technical Component available to add to your Design Studio application. It is very simple to configure, in that you simply assign it a Data Source, and you are ready to use it in your script.
Step 1: Add a Data Iterator Component
Step 2: Assign it a Data Source
Optional Event to note is 'On Data Change' - This essentially fires an event any time the Data Source changes (via filtering, navigation, etc.)
That's it. You're done configuring it. But what does it do? It's time to see what we can do with this at script-time!
Script-Time Methods
Before explaining the methods, I need to explain conceptually what it is doing with the data. Let's start with a typical Data Source's result set:
As mentioned in the use case, the first thing I do is 'flatten' the data internally into an easier to use form. This means I have some of my own conventions going forward:
- getDimensions - Returns a list of dimensions currently in Rows. This is a simple call that you can do today with DS_1.getDimensions(Axis.ROWS); but there are some subtle differences in some cases. See example below:
- getMeasures - Returns a list of items in columns. Most of the time this is usually Key Figures/Measure selection members, however it will go ahead and flatten in cases where you add a dimension in your Columns as well. This is where you would begin to struggle using standard BIAL calls to pull this off:
- getRows(optional offset, optional maxRows) - Returns rows optionally from a given row offset, and optionally a maximum amount of rows. This is where things can get interesting! Example:
So, let's consider the rows here... we have a rows.forEach loop, which returns an element (named row in my example and an index that just comes with forEach, starting at 0...) The row element itself has some BIAL methods we must explain:- getDimensionValueKey(dimensionKey) - Returns the member key for a given dimension key.
- getDimensionValueText(dimensionKey) - Returns the member text for a given dimension key.
Example:
(In this case, key and text was the same.) - getMeasureValue(measureKey) - Returns value in float format for the current row's column for the passed measureKey.
- getMeasureFormattedValue(measureKey) - Returns formatted value in String format for the current row's column for the passed measureKey.
Example:
A few things to note above. On the 3rd line, you see that we can dynamically determine in this case, the measure "key" in the first column position by saying getMeasures().atIndex(index) where 'index' indicates the column you want (beginning at 0). This is also available for getDimensions().atIndex(index). This means that you don't HAVE to hard-code a key (like we are doing with 0D_MATERIAL) and have it adapt to whatever the contents are in the result set. Also note that I switched the output from 'Formatted Text' component to my 'Rapid Prototyping' component. This is because 'Formatted Text' was stripping out certain HTML markup and I wanted to show some conditional formatting quickly.
Putting it all together:
After seeing these basic, fundamental methods, what else can you do? You can go as wild as you want! If you are accustomed to writing in languages such as JSP, BSP, ASP, etc, the use cases are endless when using Data Iterator along with Rapid Prototyping, as an example. Below are two proof-of concept examples. With some additional CSS-clean up and more time, you could come up with some super-easy to create visuals!
Simple Table showing only first 10 rows (Could enhance to paginate with buttons etc):
// Get flattened rows from Data Iterator var rows = DATAITERATOR_1.getRows(0,10); // Get dimensions (Rows) and Measures (Cols) var dimensions = DATAITERATOR_1.getDimensions(); var measures = DATAITERATOR_1.getMeasures(); // Start a simple HTML table var html = "<div style='height:400px;overflow:scroll'><table class='example'><tr>"; // Draw headers for dimensions and measures dimensions.forEach(function(element, index) { html = html + "<th>" + element.text + "</th>"; }); measures.forEach(function(element, index) { html = html + "<th>" + element.text + "</th>"; }); html = html + "</tr>"; // Loop through the rows... rows.forEach(function(row, index) { // Draw a new row html = html + "<tr>"; // Write out the dimension texts dimensions.forEach(function(member, index) { var dimText = row.getDimensionValueText(member.key); html = html + "<td class='dimension'>" + dimText + "</td>"; }); // Row striping example var stripe = "even"; if(index/2 == Math.floor(index/2)) { stripe = "odd"; } // Write out the measure formatted values measures.forEach(function(measure, index) { var measureVal = row.getMeasureValue(measure.key); var measureText = row.getMeasureFormattedValue(measure.key); html = html + "<td class='measure " + stripe +" '>" + measureText + "</td>"; }); html = html + "</tr>"; }); html = html + "</table></div>"; RAPIDPROTOTYPE_1.setHTML(html);
Simple Table/Micro Chart:
// Get flattened rows from Data Iterator var rows = DATAITERATOR_1.getRows(0,250); // Get dimensions (Rows) and Measures (Cols) var dimensions = DATAITERATOR_1.getDimensions(); var measures = DATAITERATOR_1.getMeasures(); var firstMeasureKey = measures.atIndex(0).key; // Figure out Max in BIAL for rendering chart bars var max = 0.0; rows.forEach(function(row, index) { var v = row.getMeasureValue(firstMeasureKey); if(v>max){ max = v; } }); // Start a simple HTML table var html = "<div style='height:400px;overflow:scroll'><table class='chart'><tr>"; // Draw headers for dimensions and measures dimensions.forEach(function(element, index) { html = html + "<th>" + element.text + "</th>"; }); // Draw first measure header html = html + "<th>" + measures.atIndex(0).text+"</th>"; var w = 300; html = html + "</tr>"; // Loop through the rows... rows.forEach(function(row, index) { html = html + "<tr>"; // Draw a new row dimensions.forEach(function(member, index) { var dimText = row.getDimensionValueText(member.key); html = html + "<td class='dimension'>" + dimText + "</td>"; }); var measureVal = row.getMeasureValue(firstMeasureKey); var measureFVal = row.getMeasureFormattedValue(firstMeasureKey); var barWidth = w * (measureVal / max); html = html + "<td style = 'width:" + (w+200) +";'>"; html = html + "<div style = 'display:inline-block;width:" + barWidth + "px;background-color:#006699;'> </div>" + measureFVal; html = html + "</td></tr>"; }); html = html + "</table></div>"; RAPIDPROTOTYPE_2.setHTML(html);
Runtime Example of both:
Super Goofy Scorecard Example:
// Get flattened rows from Data Iterator var rows = DATAITERATOR_1.getRows(0,10); // Get dimensions (Rows) and Measures (Cols) var dimensions = DATAITERATOR_1.getDimensions(); var measures = DATAITERATOR_1.getMeasures(); // Start a simple HTML table var html = "<table class='example scorecard'><tr>"; // Draw headers for dimensions and measures dimensions.forEach(function(element, index) { if(element.text != "Key Figures"){ html = html + "<th>" + element.text + "</th>"; } }); measures.forEach(function(element, index) { html = html + "<th>" + element.text + "</th>"; }); html = html + "</tr>"; // Loop through the rows... rows.forEach(function(row, index) { // Draw a new row html = html + "<tr>"; var priorValue = DATAITERATOR_1.makeNull(); // Write out the dimension texts dimensions.forEach(function(member, index) { var dimText = row.getDimensionValueText(member.key); if(member.text!="Key Figures"){ html = html + "<td class='dimension'>" + dimText + "</td>"; } }); // Row striping example var stripe = "even"; if(index/2 == Math.floor(index/2)) { stripe = "odd"; } // Write out the measure formatted values measures.forEach(function(measure, index) { var trend = ""; var measureVal = row.getMeasureValue(measure.key); var measureText = row.getMeasureFormattedValue(measure.key); if(index>0 && !DATAITERATOR_1.isNull(priorValue) && !DATAITERATOR_1.isNull(measureVal)){ var delta = Math.round(measureVal / priorValue * 100) + "%"; var icon = ""; if(priorValue > measureVal){ // Down trend = "downward"; }else{ // Up trend = "upward"; } measureText = "<div class='icon'></div><br />(" + delta + ")</span><br />" + measureText; } if(!DATAITERATOR_1.isNull(measureVal)){ priorValue = measureVal; }else{ priorValue = DATAITERATOR_1.makeNull(); measureText = " - "; } html = html + "<td class='measure " + stripe + " " + trend + "'>" + measureText + "</td>"; }); html = html + "</tr>"; }); html = html + "</table>"; RAPIDPROTOTYPE_3.setHTML(html);
Oh, and PS here's the CSS for my examples:
.example { border-collapse : collapse; } .example .dimension { background-color : #006699; color : #FFFFFF; } .example th { background-color : #006699; color : #FFFFFF; font-weight : bold; } .example.scorecard th { /*white-space: nowrap;*/ padding : 20px; background-color : #0099CC; color : #FFFFFF; font-weight : bold; font-size : 20pt; } .example .measure { text-align : center; } .example .icon { display : inline-block; width : 48px; height : 48px; } .example .downward .icon{ background-image : url() } .example .upward .icon{ background-image : url() } .example .downward { color : #FF0000; } .example .upward { color : #009966; } .example .measure.even { background-color : #FFFFFF; } .example .measure.odd { background-color : #DFDFDF; }
What you have seen is available for download in the usual spot (details here: SCN Design Studio 1.6 SDK Components (ver 3.0)).
Questions/Comments/Feedback always welcomed!