This best practice guide is provided to help you build applications using Canvas for TM1 that are easy to maintain and advantage of all of the Canvas features.
There are some keys principles of this document:
- Consistency IS quality!
- Readability is more important than reducing key strokes.
- All names (variables, class names, etc) should describe it's usage.
- Comment anything that isn't obvious.
- Having consistent names and case makes your code easier to follow.
- Always use the Page Creator in the /admin console to create new pages.
- It will create a HTML page and JavaScript controller.
- Use inline code for simple operations.
- Create $scope functions for more complicated code so it can be easily debugged or for reused in the page.
- Avoid the use of $root scope to pass parameters between pages.
- Use state parameters to pass simple values from one page to another.
- Update URL query parameters each time a selection is made in the page. This is important for PDF creation.
- Use Bootstrap as much as possible, it will give the look and feel consistency and allow you to switch between themes.
- Do not use JQuery, i.e.
$(".class")...
, use Angular 1 directives instead. JQuery doesn't work well wih Angular applications.
- All html elements, attributes and id's should be lower-case
- Use hyphen to separate words in a value:
id="myElement"
becomesid="my-element"
. - Use 2 spaces as indentation (the default in VS Code).
- Always indent child blocks inside a parent element.
- Use empty lines to separate logical code blocks.
- For elements with many attributes place one on each line (indented) instead of a single line.
- Always use double-quotes for values for element attributes.
- The one exception to the above rule is for
tm1-elements
. For this attribute it is better to use single quotes so elements can be qualified with a double quote.
<!-- BAD -->
<DIV ID="myELEMENT" >
</DIV>
<!-- BETTER, use lower-case and words with a hyphen -->
<div id="my-element" >
</div>
<!-- BAD -->
<div>
<ul>
<li></li>
</ul>
</div>
<!-- BETTER, indent your elements -->
<div>
<ul>
<li></li>
</ul>
</div>
<!-- BAD -->
<tm1-ui-dbr tm1-instance="Instance Name" tm1-cube="Cube Name" tm1-elements='Comma separated element list' tm1-comment-cube="Cube Name" tm1-comment-elements='Comma separated element list' tm1-picklist-radio="none" tm1-multi-line="1" tm1-read-only="false" tm1-comment-read-only="false">
</tm1-ui-dbr>
<!-- BETTER, use a single attribute per line -->
<tm1-ui-dbr
tm1-instance="Instance Name"
tm1-cube="Cube Name"
tm1-elements='Comma separated element list'
tm1-comment-cube="Cube Name"
tm1-comment-elements='Comma separated element list'
tm1-picklist-radio="none"
tm1-multi-line="1"
tm1-read-only="false"
tm1-comment-read-only="false"
>
</tm1-ui-dbr>
- Don't use
<table>
elements to layout your page. - Use Bootstrap
class="col-md-4"
classes instead, this will make your page responsive to different browser sizes. - Style all of your visual elements: tables, buttons, input, select, labels, etc with Bootstrap classes.
- Limit the use of the style attribute on elements.
- If you repeat a particular style on two or more elements make it a CSS class instead.
- Before creating your own styles try to use a Bootstrap one instead. This will allow you to change themes easily.
- Place page specific styles in a
<style>
element at the top of the page. - If you use a CSS class on more than one page place it in the
\css\style.css
- CSS class names should be lower-case and hyphenated between words:
warning-message
<!-- BAD, don't use tables for page layouts -->
<table style="width: 100%;">
<tr>
<td>
Left Column
</td>
<td>
Right Column
</td>
</tr>
</table>
<!-- BETTER, use Bootstrap grid system instead -->
<div class="row">
<div class="col-md-6">
Left Column
</div>
<div class="col-md-6">
Right Column
</div>
</div>
<!-- BAD, remember to add the Bootstrap classes -->
<table></table>
<button></button>
<input></input>
<select></select>
<!-- BETTER, add the relevant Bootstrap classes -->
<table class="table table-striped"></table>
<button class="btn btn-primary"></button>
<input class="form-control"></input>
<select class="form-control"></select>
<!-- BAD, don't repeat styles -->
<table class="table table-striped" >
<tr>
<td style="text-align: right;"></td>
<td style="text-align: right;"></td>
</tr>
</table>
<!-- BETTER, use either a Bootstrap class or a custom one -->
<table class="table table-striped" >
<tr>
<td class="text-right"></td>
<td class="text-right"></td>
</tr>
</table>
<!-- Use the style element for custom styles -->
<style>
.custom-style {
text-align: right;
font-weight: bold;
}
</style>
/* Add classes for multiple pages in css/style.css */
.custom-style {
text-align: right;
font-weight: bold;
}
/* BAD, CSS classes should be lower case */
.CUSTOMSTYLE {
text-align: right;
font-weight: bold;
}
/* BAD, separate words with a hyphen */
.customstyle {
text-align: right;
font-weight: bold;
}
/* BETTER, lower-case and hyphen as a separator */
.custom-style {
text-align: right;
font-weight: bold;
}
These JavaScript rules apply to both code in your controllers as well as inline code.
Inline code is any JavaScript inside Angular directives such as: ng-if
, ng-show
, ng-click
, etc. Example:
<!-- The code inside ng-click is just regular JavaScript -->
<!-- The semi-colons are required with writing multiple commands -->
<button class="btn btn-primary" ng-click="page.instance = 'dev'; page.cube = 'General Ledger';">Click Me!</button>
// The above code is equivalent to:
$scope.page.instance='dev';
$scope.page.cube = 'General Ledger';
- Always indent your blocks of code, i.e. if, for, etc.
- Use two spaces as indentation (the default in VS Code).
- Use empty lines to separate logical code blocks.
- Use whitespace and multiple lines to increase readability.
- Always use curly braces for all code blocks.
- All variables should use camelCase.
- In most cases a singular varaible name should be used.
- Plural should only be used in the examples below and when content of the variable is an array.
- All variables should be enclosed in an object as per Angular best practice.
- A standard set of buckets for variables are as follows:
defaults.*
are variables that are declared once and are changed in the page, otherwise known as constants in programming languages. This includes:- Static strings/values that would be hard coded in your HTML or JavaScript. i.e.
defaults.instance = "dev"
. - Variables like the current month and year that are returned from a
dbr
and used through out the page.
- Static strings/values that would be hard coded in your HTML or JavaScript. i.e.
lists.*
should be used to store any lists that are used with ng-repeat, i.e.tm1-ui-element-list
.selections.*
should be used for all selections that are made by a user in the page. Including:- Titles dimensions that are used in your page.
- Features such as zero suppression, text search and hiding/showing of sections, i.e. Q1, Q2, etc.
- Any other options that change the look and feel of the page, i.e. monthly, quarterly views. in the URL, i.e. show/hide an element in a page.
values.*
should store the result of anydbr
,dbra
or other values from server that you want to store to use elsewhere, i.e. in a calculation.
- When showing and hiding elements in a page you should indicate what it does via the name remembering that
the default value for all variables is
false
, i.e.null
is alsofalse
. Examples:- Element that defaults to hidden:
ng-if="options.showDetail"
- Element that defaults to visible:
ng-if="options.hideDetail"
- Element that defaults to hidden:
- Don't use prefixes on your variable names to indicate scope or the data type.
- All $scope level variables should be part of an object
- Code should be in the following order:
- $scope level variables
- Any functions
- Any initialisation code, i.e. requests to the server.
// BAD
$scope.ShowDetail = false;
// BETTER, variables should be camelCase and part of an object
$scope.options = {
showDetail: false
};
// BAD
$scope.defaults.products = "Test";
$scope.defaults.products = {
name: "Test"
};
// BETTER, use singular when the content is a primitive value (boolean, number or string) or an object
$scope.defaults.product = "Test";
$scope.defaults.product = {
name: "Test"
};
// BAD
$scope.defaults.month = "AUG";
// BETTER, use longer variable names that describe the contents of the variable
$scope.defaults.currentMonth = "AUG";
<!-- BAD -->
<div ng-int="ShowDetail = false"></div>
<!-- BETTER, variables should be camelCase and be part of an object -->
<div ng-int="options.showDetail = false"></div>
<!-- BAD -->
<div ng-int="defaults.month = 'AUG'"></div>
<!-- BETTER, variables should be camelCase and be part of an object -->
<div ng-int="defaults.currentMonth = 'AUG'"></div>
// BAD
if( condition ){
// No indentation
for(var i = 0; i < 10; i++){
// Do something
}
}
// BETTER
if( condition ){
// Indentation makes is easier to read!
for(var i = 0; i < 10; i++){
// Do something
}
}
// BAD, no empty lines
$scope.page = {
instance: "dev"
}
$scope.getCubes = function(){
$http.get("api/cubes/" + $scope.page.instance).then(function(success, error) {
$scope.cubes = success.data.value
});
};
$scope.dimensionNames = function(cube){
var dims = [];
_.each(cube.Dimensions, function(dim){
dims.push(dim.Name);
});
return dims.join(", ");
}
// BETTER, a line between blocks makes it easier to read
$scope.page = {
instance: "dev"
}
$scope.getCubes = function(){
$http.get("api/cubes/" + $scope.page.instance).then(function(success, error) {
$scope.cubes = success.data.value
});
};
$scope.dimensionNames = function(cube){
var dims = [];
_.each(cube.Dimensions, function(dim){
dims.push(dim.Name);
});
return dims.join(", ");
}
// BAD, always use braces and use whitespace
if( condition ) $scope.instance = "prod";
// BETTER, braces are always used and blocks should be over multiple lines
if( condition ){
$scope.instance = "prod";
}
app.controller('SampleCtrl', ['$scope', '$rootScope', '$interval', '$timeout', '$state', '$stateParams', '$http', '$location', function($scope, $rootScope, $interval, $timeout, $state, $stateParams, $http, $location) {
// Declare $scope variables first
$scope.page = {
region: "Europe",
department: "Corporate",
version: "Budget"
};
// Declare any functions
$scope.loadData = function(){
// Add load logic
};
// Call any functions
$scope.loadData();
});
One of the first questions you are going to ask when developing your first Canvas applications is when to use inline JavaScript or add page logic to the controller. Canvas was built to minimise the amount of code that needs to be written. It does most of the heavy-lifting for you, what you need to decide is whether you and fellow developers can easily maintain and follow the code.
In this section we will outline the pro's and con's of both methods. It is likely you will use both depending on the complexity of logic if it needs to be reused.
- If your logic is simple, one or two commands only. This will be the case for most pieces of logic.
- If the code isn't required in multiple locations on the page.
- If the logic is complex or requires conditional statements or loops.
- If the piece of logic needs to be reused.
- If you are having problems in your page and need to debug your logic.
- If you need access JavaScript libraries and functions.
Inline logic is added directly to the attributes (AngularJS directives) on your HTML elements such as ng-click
,
ng-if
and ng-class
. The code is just regular JavaScript that has direct access to the variables in $scope, i.e.
you do not have to prefix the variables with $scope.
. See the example below:
<!-- The code inside ng-click is just regular JavaScript -->
<button class="btn btn-primary" ng-click="page.instance = 'dev'">Click Me!</button>
You can create more complicated logic by using regular JavaScript syntax, you are however restricted to a single line. Use semi-colons to separate multiple lines of code.
<!-- The code inside ng-click is just regular JavaScript -->
<button class="btn btn-primary" ng-click="if( condition ) { page.instance = 'dev'; } else { page.instance = 'prod'; }">Click Me!</button>
The above method is the easiest method to add logic to your page and should be used as the default method. Below is a summary of the advantages and disadvantages.
Advantages | Disadvantages |
---|---|
Code is inline so it is easy to see the logic without switching to a JavaScript file | Code is placed on one line making multiple statements hard to read |
Variables that don't exist are automatically created including the hierarchy, i.e. i.e. you can just initialise page.instance = "dev" and Angular will create the page object for you. |
Variables are automatically created making it easy for a typo in a variable name to be missed |
No need to prefix variables with $scope | Most errors in your logic fail silently making it difficult to find issues |
You don't have to know a lot about JavaScript | You can't debug the code in your browser developer tools |
No direct access to JavaScript libraries such as Math |
You can also use the Angular (v1) controller for your logic. This is the best practice when developing regular Angular (v1) applications as it enables the separation of the model (data), view (HTML) and controller (logic). This is called the MVC pattern and is widely used it all types of modern applications.
Canvas applications are a little different because the directives are designed to do much of the heavy lifting for you, i.e. you don't need to create queries and request data from the server using API's. This means that you controller will be limited to more advanced scenarios such as querying the API's directly or adding complicated logic.
The controller is a JavaScript file that has the same name as your HTML file, for example /html/home.html
has a corresponding file
called /js/controllers/home.js
. Below are some examples of placing logic in controller:
<!-- Initialise the value of a scope variable -->
<!-- The 'page' object gets create automatically and the 'instance' property added to it
<div ng-init="page.instance = 'dev'"></div>
Becomes:
// Initialise the value of a scope variable
// Create the 'page' object with an 'instance' property
// Only create the 'page' object once
$scope.page = {
instance: "dev",
secondProperty: "something"
};
// The 'page' object was created above so we can now use the dot (.) notation to create a property
$scope.page.thirdProperty = "something else";
// You could also initial the 'page' variable as an empty logic
$scope.page = {};
$scope.page.instance = "dev";
$scope.page.secondProperty = "something";
<!-- Set the 'department' property on the 'page' object to 'Administration' -->
<button ng-click="page.department = 'Administration'" >Administration</button>
Becomes:
// Create a function in the controller
// This can be used multiple times on your page
$scope.setDepartment = function(department){
$scope.page.department = department;
// Add extra logic here
};
<!-- Use the setDepartment function on ng-click -->
<button ng-click="setDepartment('Administration')" >Administration</button>
<!-- You can reuse the same function again -->
<button ng-click="setDepartment('Sales')" >Sales</button>
Advantages | Disadvantages |
---|---|
Code is more readable: multiple lines, comments, etc. | Code is in a separate file so you need move back and forth to find the function |
You can reuse the logic throughout the page | JavaScript can be seen as too techie |
You have full access to the built-in JavaScript libraries and can add Angular services | |
Most errors can be seen in the JavaScript console in your browser | |
You can debug the code in your browser developer tools (press F12) |
Performance is an important consideration in any application. Canvas is designed to be easy to use, even for people with little coding background. Canvas obfuscates most of the difficult parts leaving you to focus on page design and functionality. This doesn't mean that you can just set and forget the content on your page, there is a limit to the amount of content a browser can handle quickly.
Before we outline the best practices it makes sense to describe how Canvas works and therefor it's benefits and limitations. A Canvas page is built with a collection of Angular directives that send information to the TM1 server and each other. Below is a sample of a very simple page:
<!-- A SUBMN directive is used to display a list of elements as a drop-down -->
<tm1-ui-subnm
tm1-instance="dev"
tm1-dimension="Department"
tm1-subset="Default"
tm1-default-element="6"
tm1-select-only="true"
ng-model="page.department"> <!-- Is updated when an item is selected -->
</tm1-ui-subnm>
<!-- The DBR retrieves a value from a TM1 cell and displays in on the page -->
<!-- When the page.department variable is updated in the SUBNM the DBR is refreshed -->
<tm1-ui-dbr
tm1-instance="dev"
tm1-cube="General Ledger"
tm1-elements='Actual,2011/12,Mar,Local,England,{{page.department}},Employee Benefits,Amount'
>
</tm1-ui-dbr>
The steps that turn this into a functioning page are as follows:
- The HTML source page is loaded from the server.
- The HTML is parsed by Angular so it can identify any directives in the page.
- The
tm1-ui-subnm
andtm1-ui-dbr
directives are loaded and replaced with regular HTML in the page. * Thetm1-ui-subnm
is turned into a<select>
element. * Thetm1-ui-dbr
is converted into a<span>
element. tm1-ui-subnm
loads the subset from TM1 using the REST API (via the Canvas server).tm1-ui-dbr
creates a request and adds it to the request queue.- When requests stop being added to the request queue Canvas sends a batch request to the Canvas server.
- The Canvas server converts the requests into one or more MDX statements and issues a request to the TM1 Server.
- The results from the TM1 server then are set back to the browser to be displayed in the page.
- The
tm1-ui-dbr
places a watch of thepage.department
variable waiting for any changes. 10.When a user selects an element from thetm1-ui-subnm
list thetm1-ui-dbr
recognises the change and repeats steps 5 - 8. - Angular also places watches on all of the directives waiting for any changes. See The Digest Loop and Apply
There is quite a lot going on here but it is important to understand that for every tm1-ui-dbr
on a page there is a request
and value to be retrieved using a MDX query. For every tm1-ui-*
directive the page also needs to convert the directive to
HTML. Some directives such as the tm1-ui-dbr
have complicated logic to handle comments, validation, formating, text boxes,
rich-text, etc.
What this means is that you need to think about the number of directives you place on a page. There is a limit to the number of items you can place of a page while maintaining acceptable performance. Luckily there are some easy steps you can take to manage this number and still have the functionality you expect.
Before going into specific Canvas tips we need to understand the difference between ng-if
and ng-hide
/ng-show
. Although
they seem to do the same thing how they do it is very different. ng-if
when set to false
removes the element from page (DOM)
so it no longer exists. ng-show
when false
on the other hand adds display: none
to the element so it is hidden but still
exists. Take a look at the examples below:
<!-- ng-if set to false -->
<div ng-if="false"></div>
<!-- The result in the page is just a comment -->
<!-- ngIf: false -->
<!-- ng-if set to false -->
<div ng-show="false"></div>
<!-- The result has the ng-hide CSS class added which contains: display: none !important; -->
<div ng-show="false" class="ng-hide"></div>
What this means for your pages is that elements hidden using ng-hide/show
are still present in the page and therefore
are calculated and take up resources. Elements that are hidden with ng-if="false"
no longer exist and require no calculation
or resources. When you set ng-if="true"
the elements will be dynamically loaded and updated on the fly.
- Limit the number of DBR's on a page to less than 1,000.
- Use a combination of paging and
ng-if
to limit the amount of content on a page. - Dynamic parts of a page that shown on demand should toggle
ng-if
. - Modal and dialog content should be removed with
ng-if="false"
until they are needed. - For large requests with lots of rows consider using named MDX statements and
$tm1Ui.resultsetTransform()
instead of DBRs.