Quantcast
Channel: SAP BusinessObjects Design Studio
Viewing all articles
Browse latest Browse all 662

Design Studio 1.6 SDK - Data Iterator - Read your data row by row finally!

$
0
0

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

di1.png


Step 2: Assign it a Data Source

 

di2.png

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:

 

di3.png

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:

    di4.png
  • 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:

    di5.png
  • 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:

    di6.png
    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:

    di7.png

    • 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.)
      di8.png
    • 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:

      di9.png
      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:

 

di10.png

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

 

di11.png

 

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!


Viewing all articles
Browse latest Browse all 662

Trending Articles



<script src="https://jsc.adskeeper.com/r/s/rssing.com.1596347.js" async> </script>