Challenge: we control code outside iframe and would like to trigger iframe button click
Solution: use the javascript contentWindow property, select an element and trigger the click
Iframes are useful, we can add a weather widget, map or list of recent news using external site iframe. Adding iframe to your website is easy, just copy and paste HTML code with a proper src value. What about interacting with iframe content? It’s not so easy, and sometimes not even possible, but we will try our best to show you how to use iframe click event.
Iframe modifications are possible for iframes from the same domain. There can be situations when we would like to customize iframe content or trigger some actions from outside the iframe window. In this tutorial, we will learn how to: force a button that is outside iframe – click a button placed inside iframe.
Demo with iframe
To demonstrate, we are going to create a simple HTML document with the iframe window added. Iframe will have the button: “Toggle div” that will show/hide hidden div. In the main document, we will place an additional button – “Button outside”. Our goal here is: on button click – we trigger the “Toggle div” button placed inside iframe.
The Main HTML file with iframe:
<!-- demo1.html -->
<html>
<head>
<style>
#myIframe {
height: 300px;
width:100%;
border: 2px solid #999;
}
.main {
display: flex;
flex-wrap: wrap;
}
#myButton {
margin:20px;
}
.col1 {
flex: 1;
}
</style>
</head>
<body>
<div class="main">
<div class="col1">
<iframe id="myIframe" src="iframe1.html"></iframe>
</div>
<div class="col1">
<button id="myButton" onclick="triggerClickInsideIframe()">Button outside</button>
</div>
</div>
<script>
function triggerClickInsideIframe() {
document.getElementById("myIframe").contentWindow.document.getElementById("toggleDivButton").click();
}
</script>
</body>
</html>
Content of the iframe:
<!-- iframe1.html -->
<html>
<head>
<style>
body {
background:yellowgreen;
}
#myDIV{
background:yellow;
border:3px solid brown;
padding:50px;
margin:10px 0;
}
</style>
</head>
<body>
<h1>Iframe content</h1>
<button type="button" id="toggleDivButton" onclick="toggleDiv()">Toggle div</button>
<div id="myDIV" style="display:none;">Hidden content</div>
<script>
function toggleDiv() {
var x = document.getElementById("myDIV");
if (x.style.display === "none") {
x.style.display = "block";
} else {
x.style.display = "none";
}
}
</script>
</body>
</html>
Interacting with iframe content by iframe button click event
Iframe button click event (click a button placed inside iframe) is possible by adding custom javascript in the main document. After selecting iframe, the contentWindow property returns iframe DOM that we can interact with.
An example of clicking the iframe button:
<script>
function triggerClickInsideIframe() {
document.getElementById("myIframe").contentWindow.document.getElementById("toggleDivButton").click();
}
</script>
Changing iframe content
Sometimes, we would like to change the content of iframe or hide some elements inside. We are going to add two buttons: “Change iframe background” and “Hide iframe button” that will append CSS styles to iframe.
<!-- demo2.html -->
<html>
<head>
<style>
#myIframe {
height: 300px;
width:100%;
border: 2px solid #999;
}
.main {
display: flex;
flex-wrap: wrap;
}
.btn {
margin:20px;
}
.col1 {
flex: 1;
}
</style>
</head>
<body>
<h2>Demo2</h2>
<div class="main">
<div class="col1">
<iframe id="myIframe" src="iframe1.html"></iframe>
</div>
<div class="col1">
<button id="myButton2" class="btn" onclick="changeIframeBackground()">Change iframe background</button>
<br>
<button id="myButton3" class="btn" onclick="hideIframeButton()">Hide iframe button</button>
</div>
</div>
<script>
function triggerClickInsideIframe() {
document.getElementById("myIframe").contentWindow.document.getElementById("toggleDivButton").click();
}
</script>
<script>
function changeIframeBackground(){
let head = document.getElementById("myIframe").contentWindow.document.head || document.getElementById("myIframe").contentWindow.document.getElementsByTagName('head')[0];
let style = document.createElement('style');
let css = "body{background:orange !important;}";
head.appendChild(style);
style.appendChild(document.createTextNode(css));
}
function hideIframeButton(){
let head = document.getElementById("myIframe").contentWindow.document.head || document.getElementById("myIframe").contentWindow.document.getElementsByTagName('head')[0];
let style = document.createElement('style');
let css = "#toggleDivButton {display:none}";
head.appendChild(style);
style.appendChild(document.createTextNode(css));
}
</script>
</body>
</html>
Bonus: iframe load javascript event
There is a handy on load event, which can be used to append content inside the iframe window. It is triggered when iframe is loading. If you would like to make permanent changes to iframe, here is the suggested scenario: display the animated loader, modify iframe using javascript, after completion – hide the loader.
In demo3, we are going to use the iframe load event and change the body background. In addition, we will append a custom header with the text: Appended text using JS.
<!-- demo3.html -->
<html>
<head>
<style>
#myIframe {
height: 300px;
width:100%;
border: 2px solid #999;
}
.main {
display: flex;
flex-wrap: wrap;
}
.btn {
margin:20px;
}
.col1 {
flex: 1;
}
</style>
<style>
#iframe-wrapper1.is-loading {
width:100%;
background-repeat: no-repeat;
background-position: center center;
background-image: url();
}
#iframe-wrapper1.is-loading iframe {
visibility: hidden;
}
</style>
</head>
<body>
<h2>Demo3</h2>
<div class="main">
<div class="col1">
<div id="iframe-wrapper1" class="is-loading">
<iframe id="myIframe" src="iframe1.html"></iframe>
</div>
</div>
</div>
<script>
document.getElementById('myIframe').onload = function() {
let head = document.getElementById("myIframe").contentWindow.document.head || document.getElementById("myIframe").contentWindow.document.getElementsByTagName('head')[0];
let body = document.getElementById("myIframe").contentWindow.document.body || document.getElementById("myIframe").contentWindow.document.getElementsByTagName('body')[0];
let style = document.createElement('style');
let css = "body{background:red !important;} #toggleDivButton {display:none}";
let header = "<h1 style='color:yellow'>Appended text using JS</h1>";
head.appendChild(style);
style.appendChild(document.createTextNode(css));
body.innerHTML += header;
setTimeout(function(){
document.getElementById("iframe-wrapper1").classList.remove("is-loading");
},2000);
};
</script>
</body>
</html>
Cross-origin frame
What about using iframe, but from a different domain? For example, a Wikipedia article. Let’s try it:
<div id="iframe-wrapper1">
<iframe id="myIframe" src="https://en.wikipedia.org/wiki/Sociable_weaver"></iframe>
</div>
Iframe onload displays the following error:
Uncaught DOMException: Blocked a frame with origin “http://localhost:63342” from accessing a cross-origin frame. at document.getElementById.onload
What is the reason? Same-origin policy which is a critical security mechanism. It prevents a malicious website from running Javascript code on a third-party website. Two URLs have the same origin if the protocol and host are the same for both. If the port if specified, it also should match.
To sum up, when we load iframe from the same domain, we can modify it, otherwise – we cannot.
Source code of javascript iframe events
The demos presented in this tutorial can be downloaded from github. Make sure to load files using the server (localhost or normal server), otherwise iframe may not work. Repository: https://github.com/createit-dev/166-how-to-click-programmatically-button-inside-iframe
That’s it for today’s tutorial. Subscribe to our newsletter to stay up-to-date with other useful tips and guidelines.
Do you need someone to implement this solution for you? Check out our specialists for hire in the outsourcing section.
Are you considering a global project and are uncertain how to proceed? See our custom web application development section and see how we do it!
Comments
0 response