One of the many challenges developers face when building applications is usability, and a common problem is making it obvious that a user needs to scroll to see more data.
This problem can be compounded when embedding content from any external application; even Sigma.
Scroll bars on both parent and iframe content can significantly impact the user experience, often in a negative way. Here's why:
To avoid these issues, it's generally best to avoid having scroll bars on both the parent and the iframed content. This can often be achieved by making the iframe (and potentially also the parent page) responsive, so that the iframe adjusts its size based on the content it contains and the size of the viewport.
In essence, making an iframe dynamic or responsive is about ensuring that the content within the iframe is accessible, user-friendly, and looks good on all devices and window sizes. It's about creating a better, more consistent user experience.
Since Sigma uses iframes for embedding, it is fairly easy to avoid this situation.
This QuickStart assumes you have already taken the QuickStart Embedding 1: Prerequisites so that you have a sample environment to complete the tasks in this QuickStart.
Developers who are interested in how to leverage dynamic iframes to embed Sigma into their applications.
Click here to download sigma_embed_responsive.zip
The zip file contains these two files:
index.html: the web page that contains the iframe we are embedding into. No changes are required for this file.
embed-api.js: a JavaScript routine that sets up the services required and configuration of the Sigma options. This is a example of an Embed API
.
If you haven't installed Node already, please do so by referring to section 3 of the QuickStart: Embedding 01: Prerequisites.
If you have already installed Node, recall that we still need to install the required Node packages for our new sigma_embed_responsive
folder that was created when we unzipped the download into the sigma_embedding
folder.
Open a new Terminal session from the folder sigma_embed_drs
and run this command:
Run the command tro install the Express web-server:
npm init
As in the prerequisites QuickStart, accept all the defaults by pressing enter until completed.
and...
Run the command:
npm install supervisor
Login to Sigma and navigate to the Templates
page.
Click to select the Plugs Electronics Profit Planning Tool
template:
Dismiss
the dialogue asking if you want to use your own data.
Click the Save As
button and name the Workbook Dynamic iframes
.
Share
the Workbook with the Sales_Managers
team we created in the earlier QuickStarts.
Give the team Can view
permission.
Open the Embedding
controls:
Select Profit Planning Tool
and select Copy
:
In the node project folder, open embed-api.js
and replace the value for EMBED PATH
with this new value.
Ensure that embed-api.js
has the correct API and Embed Secrets.
Once embed-api.js
is setup, make sure that node's express server is running:
Start Terminal and run:
supervisor embed-api.js
Browse to localhost:3000 to verify that we have a working application with our Sigma table on it.
When you reduce the size of the browser you can see that there are two scroll bars present.
This is what we want to try and avoid by making our iframe dynamic.
Since we are embedding content that is created in Sigma, it is not always possible to ensure that the Sigma content will "fit" inside a static iframes dimensions, since the content may vary depending on use-case.
Without control over the iframe's content, it's challenging to have a fully dynamic solution that adjusts to the exact height of the content every time.
Given the initial constraint, one approach might be to:
1: Set a default height for the iframe that should ideally fit most of the content you expect to embed.
2: Allow the iframe to scroll if its content exceeds this default height.
This ensures that:
1: The parent application remains without a scrollbar, providing a cleaner overall look.
2: The iframe will only show a scrollbar when absolutely necessary (i.e., when its content exceeds its set height).
This approach provides somewhat of a balance between user experience and practicality, given the constraints, but does not fully address the issue of scroll bars.
In this scenario, even when the browser is full screen, the embedded content we have elected to provide from Sigma will be too long, and thus the iframe requires a scroll bar.
The parent page, however, should still avoid having a scroll bar.
For this example, we will simply allow the iframe to scroll, but ensure the parent page does not.
Replace the code in your embed project folder for index.html
with this code:
<!DOCTYPE html>
<html>
<head>
<title>Sigma Embedding - Application</title>
<style>
/* Ensure html and body tags occupy full height */
html, body {
height: 100%; /* This makes sure the entire height of the viewport is covered by the page. */
margin: 0; /* Resetting default margins to prevent any unexpected space. */
padding: 0; /* Resetting default padding. */
overflow: hidden; /* This hides scrollbars and prevents scrolling on the parent page. */
}
/* Set a height for the iframe and allow it to scroll */
#sigmaDashboard {
height: 85vh; /* This sets the iframe height to be 85% of the viewport's height. */
width: 100%; /* This ensures the iframe takes the full width of its container. */
overflow: auto; /* This allows scrolling within the iframe if its content exceeds its height. */
}
</style>
</head>
<body>
<h2>Sigma Embedding - Application</h2>
<h3>iframe URL below comes from API call to embed-api.js</h3>
<iframe id="sigmaDashboard"></iframe> <!-- This is the iframe that will embed the content. -->
<script>
/* Define the API URL from where the iframe's source will be fetched. */
const URL = "http://localhost:3000/api/generate-embed-url";
/* Fetch the URL using the Fetch API */
fetch(URL)
.then(data => {
return data.json(); /* Parse the response data as JSON. */
})
.then(res => {
/* Set the iframe's source URL to the value fetched from the API. */
document.getElementById("sigmaDashboard").src = res.url;
})
.catch(e => {
/* Log any errors that occur during the fetch. */
console.log(e);
});
</script>
</body>
</html>
Once you have made the changes and saved index.html
, refresh the browser page.
Now we only have one scroll bar (in the embed) and the parent application does not present a scroll bar regardless of how we resize the browser.
The calc() function in CSS can be used to dynamically compute values for various properties. While calc() can be powerful in many scenarios, it's important to understand its limitations and its applicability to your specific case.
In our example, the calc() function can be used to compute dimensions based on various units, combinations of units, or viewport sizes.
For example, if we want the iframe to take up the full viewport height minus some space for a header, you could use calc():
#sigmaDashboard {
height: calc(100vh - 60px); /* Assuming 60px is the height of your header */
width: 100%;
overflow: auto;
}
This would make the iframe's height equal to the full viewport height minus 60 pixels for the header.
To test this, we simple replace the relevant code for #sigmaDashboard
. Here is that code for your convienence.
<!DOCTYPE html>
<html>
<head>
<title>Sigma Embedding - Application</title>
<style>
/* Ensure html and body tags occupy full height */
html, body {
height: 100%;
margin: 0;
padding: 0;
overflow: hidden; /* To prevent scrolling on the parent */
}
/* Set a default height for the iframe and allow it to scroll if content exceeds */
#sigmaDashboard {
height: calc(100vh - 60px); /* Assuming 60px is the height of your header */
width: 100%;
overflow: auto;
}
</style>
</head>
<body>
<h2>Sigma Embedding - Application</h2>
<h3>iframe URL below comes from API call to embed-api.js</h3>
<iframe id="sigmaDashboard"></iframe>
<script>
const URL = "http://localhost:3000/api/foo";
fetch(URL)
.then(data => { return data.json() })
.then(res => { document.getElementById("sigmaDashboard").src = res.url })
.catch(e => console.log(e));
</script>
</body>
</html>
In summary, while calc() is a valuable tool in CSS for dynamic calculations, it doesn't immediately solve the scrollbar issues.
It can aid in setting dimensions based on other known dimensions or viewport sizes, but it won't "dynamically" adjust the iframe's height based on its content.
Sigma provides a JavaScript method that enables developers to dynamically adjust the iframe height, based on the content's actual height.
This is especially useful when the content inside the iframe can change dynamically, leading to different height requirements.
This can be used to fully address the scroll bar issues.
The method is workbook:pageheight:onchange
, and is an event
in Sigma.
You can read about Javascript actions in user-backed embeds here.
We can use this event to adjust the iframe's height in real-time, ensuring that the iframe always matches the height of its content.
Here's are the steps required to integrate the workbook:pageheight:onchange
event, along with sample code:
1: Listen for the Event:
You'll need to add an event listener to the window object that listens for the message event. This is because cross-document communication (like between a parent page and an iframe) uses the postMessage method and message event.
2: Adjust the iframe Height:
When the event is received, we check if its type matches ‘workbook:pageheight:onchange' and then adjust the iframe height based on the pageHeight value provided.
3: Modified code:
Here's the modified script to include the handling of the workbook:pageheight:onchange event:
<!DOCTYPE html>
<html>
<head>
<title>Sigma Embedding - Application</title>
<style>
/* Ensure the entire height of the viewport is covered by the page. */
html, body {
height: 100%;
margin: 0; /* Reset default margins to prevent any unexpected spacing. */
padding: 0; /* Reset default padding. */
overflow: hidden; /* Prevent scrolling on the parent page. */
}
/* Set default dimensions for the iframe. */
#sigmaDashboard {
height: 85vh; /* Set the iframe height to be 85% of the viewport's height. */
width: 100%; /* Ensure the iframe takes the full width of its container. */
overflow: auto; /* Allow content inside the iframe to scroll if it exceeds the iframe's height. */
}
</style>
</head>
<body>
<h2>Sigma Embedding - Application</h2>
<h3>iframe URL below comes from API call to embed-api.js</h3>
<iframe id="sigmaDashboard"></iframe> <!-- The iframe that will embed content. -->
<script>
/* Define the API URL from which the iframe's source will be fetched. */
const URL = "http://localhost:3000/api/foo";
/* Fetch the URL using the Fetch API. */
fetch(URL)
.then(data => {
return data.json(); /* Parse the fetched data as JSON. */
})
.then(res => {
const iframe = document.getElementById("sigmaDashboard");
iframe.src = res.url; /* Set the iframe's source URL to the value fetched from the API. */
/* Listen for messages (events) from the iframe content. */
window.addEventListener('message', function(event) {
/* Check if the received event is of type 'workbook:pageheight:onchange'. */
if (event.data && event.data.type === 'workbook:pageheight:onchange') {
/* Adjust the iframe's height based on the height of its content. */
iframe.style.height = event.data.pageHeight + 'px';
}
});
})
.catch(e => {
/* Log any errors that occur during the fetch operation. */
console.log(e);
});
</script>
</body>
</html>
In the sample code:
After updating our HTML page to use this code, we see the following results (using Safari's developer tools to simulate different device screen sizes):
In this lab we learned how to implement a responsive iframe Sigma embed into an html page, using CSS and Javascript methods.
Choosing the best approach is up to the developer, but we hope this QuickStart provided useful information when embedding Sigma.
Additional Resource Links
Be sure to check out all the latest developments at Sigma's First Friday Feature page!