How can I make an iframe click-through, but make that iframe's body still be clickable?
I tried:
iframe.style.width = '100%'
iframe.style.height = '100%'
iframe.style.display = 'block'
iframe.style.position = 'fixed'
iframe.style.backgroundColor = 'transparent'
iframe.style.pointerEvents = 'none'
iframe.style.border = '0'
iframe.frameborder = '0'
iframe.scrolling = 'no'
iframe.allowTransparency = 'true'
and inside of my I frame I'm using the following css:
html, body {
/* background:none transparent; */
pointer-events:auto;
}
This results in body being visible (which is what I want), but it is click-through like the rest of the iframe. I want the body of the iframe to be clickable, but all the rest of the actual iframe element should be click-through.
The iframe is always bigger than the body inside of it.
Unfortunately I cannot access the the iframe content from the main site (so accessing the scrollHeight etc isn't possible), I can only change its actual source code.
How can I make an iframe click-through, but make that iframe's body still be clickable?
I tried:
iframe.style.width = '100%'
iframe.style.height = '100%'
iframe.style.display = 'block'
iframe.style.position = 'fixed'
iframe.style.backgroundColor = 'transparent'
iframe.style.pointerEvents = 'none'
iframe.style.border = '0'
iframe.frameborder = '0'
iframe.scrolling = 'no'
iframe.allowTransparency = 'true'
and inside of my I frame I'm using the following css:
html, body {
/* background:none transparent; */
pointer-events:auto;
}
This results in body being visible (which is what I want), but it is click-through like the rest of the iframe. I want the body of the iframe to be clickable, but all the rest of the actual iframe element should be click-through.
The iframe is always bigger than the body inside of it.
Unfortunately I cannot access the the iframe content from the main site (so accessing the scrollHeight etc isn't possible), I can only change its actual source code.
Share Improve this question edited Jan 20, 2017 at 16:58 Forivin asked Jan 20, 2017 at 14:33 ForivinForivin 15.6k30 gold badges118 silver badges212 bronze badges 10- 1 If I got it right, you want the iframe element to be clickable, but nothing inside it? – fingeron Commented Oct 31, 2018 at 9:03
- what click-through means? click with no event? – SilentTremor Commented Oct 31, 2018 at 9:03
- @fingeron What I assume he was asking, was is it possible to have the HTML inside of the iFrame catch a mouse event, however if no HTML elements exist where the cursor fires in the iFrame, the mouse event will pass through the iFrame to the HTML on the original DOM. – Ian Wise Commented Oct 31, 2018 at 9:05
- @SilentTremor 'click through' refers to not catching a mouse event, as if the element does not exist. – Ian Wise Commented Oct 31, 2018 at 9:07
- 1 Ian Wise is correct with both, his assumption what I was asking and with what 'click through' means in this context. – Forivin Commented Nov 5, 2018 at 10:21
2 Answers
Reset to default 7 +100DISCLAIMER: OP created this question almost two years ago, my answer follows Ian Wise's bumping the question and elaborating on it (see ments).
What you are describing here involves logic between a document and a child document: "If a click event did nothing inside child document, apply that click event to parent document", and therefore cannot be approached using HTML/CSS.
Iframes are different documents. They do have a child-parent relationship with their containers, but an event that occurs within the iframe will be handled by the iframe.
An idea that requires some code but will work:
- Place a transparent div above all the stacked iframes, and catch the click event pos.
- Parent logic ->
- Iterate through array of existing iframe elements.
- Send click pos until one of the iframes returns a positive response.
function clickOnCover(e, i) {
if(e && e.preventDefaule) e.preventDefault();
if(i && i >= iframes.length) {
console.log("No action.");
return;
}
var iframe = iframes[i || 0];
if(iframe.contentWindow && iframe.contentWindow.postMessage) {
iframe.contentWindow.postMessage({ x: e.clientX, y: e.clientY, i: i });
}
}
function iframeResponse(e) {
var response = e.data, iframeIndex = response.i;
if(response.success)
console.log("Action done on iframe index -> " + iframeIndex);
else
clickOnCover({ clientX: response.x, clientY: response.y }, iframeIndex+1);
}
- iFrames logic ->
- Have a function that accepts the
clientX, clientY
and checks for possible activies in that position (might be tricky!). - Will respond positively if an action occurred, and the opposite.
- Have a function that accepts the
window.addEventListener("message", function(e) {
// Logic for checking e.x & e.y
e.success = actionExists; // Some indicator if an action occurred.
if(window.parent && window.parent.postMessage) {
window.parent.postMessage(e);
}
});
This solution keeps managing the event within the parent document and only requires iterating through whatever amount of stacked iframes.
Found a relevant SO question to further support my claim: Detect Click in Iframe
This is not possible with CSS.
The easiest way is to resize the iframe properly. Assuming you have access to iframe content, the following solution is possible:
You might add a little JS to allow the parent page to know the iframe height
mainpage.js
var iframe = getIframe();
setIntervalMaybe(() => {
// ask for size update
iframe.contentWindow.postMessage({action: "getSize"}, document.location.origin);
}, 100)
window.addEventListener("message", function receiveMessage(event) {
switch(event.data.action) {
case "returnSize":
// updateIFrameSize(event.data.dimensions);
console.log(event.data.dimensions);
break;
}
}, false);
iframe.js
window.addEventListener("message", function receiveMessage(event) {
switch(event.data.action) {
case "getSize":
event.source.postMessage({action: "returnSize", dimensions: {
width: document.body.offsetWidth,
height: document.body.offsetHeight
}}, event.origin);
break;
}
}, false);