Skip to content

Widgets

Simple table

In a simple table, you can aggregate your data with as many dimensions as your want and display the corresponding measures. You can also add computed measures. A computed measure ia a column that is obtained as a formula using other columns content. In the screenshot below (which is a folded table), the column average basket is a computed measure obtained with the formula: line[4]/line[3].

Folding

If you enable the folding option (in the rows folding menu), the user can click on the ± icons in the first column:

alt text

In a folded table, the blue row is the grand total, and pink rows are intermediate totals. The grand total is only displayed if the option Display total is enabled for at least one column. To hide the intermediate total of a column, use the column formatting option Hide intermediate total.

KPI widgets

A KPI widget is used to show high-level KPIs (e.g. current year revenue, percentage value...). You can change the template to change the rendering of the widget.

List of icons

In some templates, you can choose an icon to be show in the widget. The list of available icons is available here.

Free table

The free table lets you configure cells one by one. Each cell can be of mode:

  • text: the cell content is a simple text.
  • value: the cell content is obtained from one datasource query.
  • formula: the cell content is a formula based on other cells.

HTML mode

When the HTML mode is enabled for a cell, the HTML code provided will be rendered into this cell.

When a cell is in value mode, {{valueN}} can be used to obtain the Nth value configured in the data menu. Note that the Computing formula and the Numeral formatting are already applied to the value.

It's also possible to use simple expressions such as: {{value2>=0 ? "green" : "red"}} or to use any dashboard function.

Using values and numeral formatting

If you apply a numeral formatting (e.g. with option "12.34%" to format the value as a percentage), the corresponding {{valueN}} returns a string including the percentage sign. If you write {{valueN*2}} or {{valueN >0 ? "a" : "b"}} an error will be returned.

In this case, you need to convert valueN to a number. To do that, use the toNumber function, e.g.: {{toNumber(valueN)*2}}.

Example:

<div style="background-color:#fafafa;
            color:black;
            height:100%;
            text-align:left;
            padding-left:1vw;
            padding-top:1vw;
            padding-bottom:1vw;
            display:flex;
            flex-direction: column;
            justify-content:space-around">

    <div style="color:rgb(77, 144, 147); font-size:1.4vw">
        Turnover
    </div>

    <div style="font-size:2.5vw; color:rgb(0, 96, 100)">
        {{value1}}
    </div>

    <div style="font-size:1.2vw; color: {{value2>=0 ? "green" : "red"}}">
        <span class="fa fa-angle-double-{{value2>=0 ? "up" : "down"}}"></span>
        {{value2}} vs Month-1
    </div>
</div>

Buttons

This widget allows you to add buttons into your dashboard. Each button can either:

  • Trigger an automation (of any type)
  • Download the full content of a Storage
  • Download a static file (i.e. a file of any type (pdf, xlsx, ppt...) that you previously uploaded in Serenytics)
  • Set a variable's value
  • Open a link
  • Open a tab
  • Execute Javascript code

In the studio, you cannot click on a button to trigger its action (as the click on a widget is used to select a widget). If you want to trigger the action of a button, you need to click on Simulate click in the button's settings menu.

Trigger an automation

To enable an automation to be run by other users than yourself, you need to enable the option named All all users to run this script. This option is in the Advanced tab of the automation's settings. If you don't enable this option on the script, its execution from a dashboard will return an error. This is a security setting to ensure that by default, an automation cannot be run by all users.

When a button linked to an automation is clicked, the automation is put in the execution queue in the server. Once there is some resource available to execute it, it will be executed. In most cases, a resource is available immediately. But in some occasions, the user might have to wait for the execution to start.

Once an automation is started, the user cannot stop it. Even if he closes the task window, the automation will continue its execution.

An automation can only have one instance running. If an automation is triggered, and then it is triggered again while it is already running, the second execution will return an error message. This can happen if you share a dashboard with buttons to multiple users and they click on a button triggering an automation at the same time.

Pass parameters to a Python script

In the settings of a button, you can specify a JSON. If your automation is a Python script, this JSON can be retrieved in the Client object:

Example:

import serenytics
client = serenytics.Client()

params = client.script_args
print(params)

Variables from dynamic filters and input fields

If you have created dynamic filters or input fields that store their values into variables, these values are also passed to the python scripts triggered by a dashboard button. They can also be retrieved in the script_args field. This is a simple way to ask parameters to a user in a dashboard and then have a button executing a Python script using these parameters.

Download the full content of a storage

This type of button only works for Storage datasource. When the button is clicked, a popup will ask the user for some file settings (csv separator, quotes).

Download a static file

With this option, when the user clicks on the button, a file will be downloaded. The files to be downloaded must be previously uploaded in the Admin section, in the Static File tab.

There are two modes to specify the file to download:

  • Static: you need to select the filename within the list of your files.
  • Dynamic: you can dynamically build the filename using variables, for example: my-report-{{varYear}}.xlsx where varYear is a variable in your app.

Set a variable's value

A click on the button will set the variable to a given value.

A click on the button will open a new URL. You can specify if you want this new URL to be opened in the same window or in a new one (default mode).

You can use variables in your URL

For example, you can use https://www.google.com/search?q={{myVariable}}. This can be used to redirect the user to a given page in another webapp, using a URL such as: https://www.myApp.io/products/{{productId}}

Open a tab

A click on the button will open another tab from the app. You must provide the tab index (1 is for the first tab).

This also works if your tabs are hidden

When the Show tabs option is disabled in the Layout & Print menu, these buttons will still work. So you can create your own navigation menu using these buttons and disabling the default tabs buttons.

Execute Javascript Code

A click on the button will execute the provided Javascript code.

You can use the helpers provided by Serenytics to have interactions with your app: Javascript helpers

Map

By default, the popup (when you click on a marker) displays the Label column and a list of the selected measures. You can configure what is displayed in the popup by filling in the Popup HTML template field.

The HTML code provided can contain formulas inside double curly brackets: {{line[1]}}. The available variables are:

  • value: contains the value of the Label column
  • line[N]: contains the value of the Nth selected measure column

Example:

<h4>{{value}}</h4>

<strong>Population: </strong>{{line[1]}}
<br>
<strong>Altitude : </strong>{{line[2]}}

Custom widget

The "custom" widget lets you code your own widgets. The simplest widgets use only HTML and CSS. For more advanced widgets (mostly widgets with interaction such as filters), you can also use Javascript (see here for a list of helpers to add interactions in your widgets).

Retrieving data in the widget code

In the data configuration of the widgets, configure the dimensions and metrics like for any other widget.

Then, in the HTML code, you can retrieve this data using the variables:

  • data_rows: contains the list of rows (use data_rows[1][1] to get first value of first row)
  • data_headers: contains the list of headers
  • data_nb_rows: contains the number of rows in the dataset
  • data_row_length: contains the length of each row (i.e. this is also the number of headers)

So the simplest widget to show these values would be:

<div>
    Rows: {{data_rows}}
</div>
<div>
    Headers: {{data_headers}}
</div>
<div>
    Nb rows: {{data_nb_rows}}
</div>
<div>
    Length of each row: {{data_row_length}}
</div>

The {{xxx}} syntax means the content is interpreted by the Serenytics interpreter. This interpreter has access to the four variables described above. But it also has access to all the dashboard variables (and filters' variables). And all the dashboard formulas are also available.

You can use condition using the syntax:

{{myCondition ? value_if_true : value_if_false}}.

Create a library of custom widgets using Snippets

Usually, you want to re-use the code of a custom widget in several dashboards. To do that, write your widgets' code in Snippets and reference thme in your custom widgets in dashboards.

See the full Snippets documentation. We strongly encourage you to read the Snippet documentation as there are many explicit examples.

Older (and obsolete) version

In previous version your could retrieve the data using the strings {{rows}} and {{headers}} in your code. It is still working but we advise to use the new variables.

Example:

<script>
    var rows = JSON.parse('{{rows}}');
    var headers = JSON.parse('{{headers}}');

    console.log('data rows:', rows);
    console.log('data headers:', headers);
</script>

If you have a data set where you selected two headers Country and Amount, the following code will be executed:

<script>
    var rows = JSON.parse('[["France",470], ["USA":2550]]');
    var headers = JSON.parse('[{"name":"Country","type":"str"},{"name":"Amount","type":"float"}]');

    console.log('data rows:', rows);
    console.log('data headers:', headers);
</script>

Example to show a simple table

Here is an example that will get the data configured in your widget and show it in a simple table.

<div id="customWidget" style="overflow: auto; height: 100%;"></div>

<script>
    function renderTable() {
        // --- Retrieve data from Serenytics template injection
        var rows = JSON.parse('{{rows}}');
        var headers = JSON.parse('{{headers}}');

        if (!rows) {
            console.log('data is not yet loaded');
            return;
        }


        var table = document.createElement('table');
        table.style.width = '100%';
        table.style.borderCollapse = 'collapse';

        var thead = document.createElement('thead');
        var headerRow = document.createElement('tr');
        headers.forEach(header => {
            var th = document.createElement('th');
            th.textContent = header.name;
            th.style.border = '1px solid #ddd';
            th.style.padding = '8px';
            th.style.textAlign = 'left';
            th.style.backgroundColor = '#f2f2f2';
            headerRow.appendChild(th);
        });
        thead.appendChild(headerRow);
        table.appendChild(thead);

        var tbody = document.createElement('tbody');
        rows.forEach(row => {
            var tr = document.createElement('tr');
            row.forEach(cell => {
                var td = document.createElement('td');
                td.textContent = cell;
                td.style.border = '1px solid #ddd';
                td.style.padding = '8px';
                tr.appendChild(td);
            });
            tbody.appendChild(tr);
        });
        table.appendChild(tbody);

        var container = document.getElementById('customWidget');
        container.appendChild(table);
    }

    renderTable();

</script>

<style>
    #customWidget table {
        font: 12px sans-serif;
        vertical-align: middle;
        text-align: left;
        border-collapse: collapse;
    }

</style>

Example to show a table with a popup when a value is clicked

In this example, the third column will have a link to open a popup instead of the cell's value. The popup will show the value.

<div id="customWidget" style="overflow: auto; height: 100%;"></div>

<script>
    function showCustomPopup(message) {
        // Create a popup element
        var popup = document.createElement('div');
        popup.textContent = message;
        popup.style.position = 'fixed';
        popup.style.top = '50%';
        popup.style.left = '50%';
        popup.style.transform = 'translate(-50%, -50%)';
        popup.style.padding = '20px';
        popup.style.backgroundColor = 'white';
        popup.style.border = '1px solid #ddd';
        popup.style.borderRadius = '10px';
        popup.style.boxShadow = '0px 4px 8px rgba(0, 0, 0, 0.2)';
        popup.style.zIndex = '1000';

        // Add a close button
        var closeButton = document.createElement('button');
        closeButton.textContent = 'Close';
        closeButton.style.marginTop = '10px';
        closeButton.className= 'btn btn-primary'
        closeButton.onclick = function () {
            document.body.removeChild(popup);
        };

        var space = document.createElement('div');
        popup.appendChild(space);
        popup.appendChild(closeButton);
        document.body.appendChild(popup);
    }

    function renderTable() {
        // --- Retrieve data from Serenytics template injection
        var rows = JSON.parse('{{rows}}');
        var headers = JSON.parse('{{headers}}');

        if (!rows) {
            console.log('data is not yet loaded');
            return;
        }


        var table = document.createElement('table');
        table.style.width = '100%';
        table.style.borderCollapse = 'collapse';

        var thead = document.createElement('thead');
        var headerRow = document.createElement('tr');
        headers.forEach(header => {
            var th = document.createElement('th');
            th.textContent = header.name;
            th.style.border = '1px solid #ddd';
            th.style.padding = '8px';
            th.style.textAlign = 'left';
            th.style.backgroundColor = '#f2f2f2';
            headerRow.appendChild(th);
        });
        thead.appendChild(headerRow);
        table.appendChild(thead);

        var tbody = document.createElement('tbody');
        rows.forEach((row, rowIndex) => {
            var tr = document.createElement('tr');
            row.forEach((cell, colIndex) => {
                var td = document.createElement('td');
                td.textContent = cell;
                td.style.border = '1px solid #ddd';
                td.style.padding = '8px';

                // Add click event for the popup
                if (colIndex ===2) {
                    td.style.cursor = 'pointer'; // Add pointer cursor to indicate interactivity
                    td.textContent = "Résumé";
                    td.onclick = function () {
                        showCustomPopup(`${cell}`);
                    };
                }

                tr.appendChild(td);
            });
            tbody.appendChild(tr);
        });
        table.appendChild(tbody);

        var container = document.getElementById('customWidget');
        container.appendChild(table);
    }

    renderTable();

</script>

<style>
    #customWidget table {
        font: 12px sans-serif;
        vertical-align: middle;
        text-align: left;
        border-collapse: collapse;
    }

</style>

Loading Javascript or CSS libraries

If you use one or more 3rd party javascript librairies, you need to load them with our helper function serenyticsLoadAllScripts(list_of_urls, callback) as shown in the example using D3.js below.

Using third party javascript or CSS libraries

Of course you can use 3rd party libraries such as D3.js to code widgets exactly as you want.

However, you should only load libraries that are not already used by the Serenytics application, such as D3.js or Highcharts. You should not try to load a different version of a library already used by Serenytics (such as ChartJs, Angular, Lodash, and more). You should not either load a different version of a CSS library already used by Serenytics, such as Bootstrap.

If you are not sure, please contact Serenytics support. Otherwise your widget could break in a future upgrade of the Serenytics platform.

Complete example using D3.js

<!--
    For this example to work, you need to select 2 columns in the Data config menu.
    The first column will be used as labels and the second one as numerical values.
-->

<div id="customWidget" style="overflow: auto; height: 100%;"></div>

<script>
    function main() {
        // --- Retrieve data from Serenytics template injection
        var rows = JSON.parse('{{rows}}');
        var headers = JSON.parse('{{headers}}');

        if (!rows) {
            console.log('data is not yet loaded');
            return;
        }

        // --- Create a table with inline bars using d3.js

        var table = d3.select("#customWidget").append("table");

        // Create a table with rows and bind a data row to each table row
        var tr = table.selectAll("tr.data")
            .data(rows)
            .enter()
            .append("tr")
            .attr("class", "datarow");

        // Apply style to even rows
        d3.selectAll(".datarow").filter(":nth-child(even)").attr("class", "datarow even")

        // Create the label column
        tr.append("td").attr("class", "data name")
            .text(function (d) {
                return d[0];
            });

        // Create the value column
        var formatValue = d3.format(".2f");
        tr.append("td").attr("class", "data value")
            .text(function (d) {
                return formatValue(d[1]);
            })

        // Create the chart column
        var chart = tr.append("td").attr("class", "chart");

        // Create the div structure of the chart
        chart.append("div").attr("class", "bar");

        // Create the scale to compute bar width
        var xScale = d3.scaleLinear()
            .domain([0, d3.max(rows, function (d) {
                return d[1];
            })])
            .range(["0%", "100%"]);

        // Creates the chart bars
        tr.select("div.bar")
            .style("width", "0%")
            .transition()
            .duration(500)
            .style("width", function (d) {
                return xScale(d[1]);
            });
    }

    serenyticsLoadAllScripts(
        ["https://cdnjs.cloudflare.com/ajax/libs/d3/4.7.3/d3.min.js"],
        main
    );
</script>

<style>
    #customWidget .chart {
        width: 300px;
        padding-left: 20px;
    }

    #customWidget .bar {
        height: 17px;
        float: left;
        background-color: steelblue;
    }

    #customWidget table {
        font: 14px sans-serif;
        vertical-align: middle;
        text-align: left;
        border-collapse: collapse;
    }

    #customWidget td,th {
        padding-top: 2px;
        padding-bottom: 2px;
    }

    #customWidget tr.even {
        background-color: whitesmoke;
    }

    #customWidget th {
        padding-left: 10px;
    }

    #customWidget td.data {
        padding-left: 10px;

    }
    #customWidget td.value {
        text-align: right;
    }
</style>