In the component template section of an Astro page, I am adding the following script:
// src/pages/foo.astro
<script>
document.addEventListener('DOMContentLoaded', function() {
const myElement = document.getElementById('myId');
if (myElement) {
myElement.addEventListener('click', function() {
console.log('Clicked on element with ID "myId"');
// Add your desired functionality here
});
} else {
console.warn('Element with ID "myId" not found');
}
});
</script>
It's a generic script attaching a click event handler to myId
element if found.
Note that the event is attached when the DOM content is loaded, via the DOMContentLoaded
listener, to be sure that myId
is created when due.
If I navigate directly to the page, I can read the messages sent by console.log()
, depending on the presence of myId
.
However, linking to foo.astro
from another page:
// src/pages/bar.astro
...
<a href="/foo.astro">Go to foo</a>
then, when I click on the link to foo
, the DOMContentLoaded
event is not fired and I see no console messages.
This happens also when I look for elements with different functions, such asdocument.querySelectorAll()
.
So, what is the correct way to add event handlers in Astro?
Should I avoid waiting for DOM loading and expect that Astro will inject my page scripts when it's the right time? And how is this decided by Astro?
Are you using Astro ViewTransition
s? If so, that would explain the difference in behaviour. And you should listen to the astro:page-load
event.
btw. <script>
tags in Astro without any attributes are converted to <script type="module">
and thus you don't need any DOMContentLoaded
listener (see Astro docs). JS Modules are only executed once the DOM is loaded.