The error message which you are receiving means that at the point when this code is executed, there is no element rendered in the browser that can be found with the id of menuLink. When you manually execute the statement in the browser console, you're doing so after the entire page has been rendered and menuLink is now in the page context. You will probably find that this script is executing before the DOM is fully loaded even though you've got it at the end of the VF markup. (What tags did you use to do this? <apex:includeScript
?)
If you're using <apex: id="yourName"
tags for these elements now or in the future, the following may be useful.
Salesforce documentation about accessing the Id as it was rendered in the DOM.
Best Practices for Accessing Component IDs
You don't mention where this ui.js
file is stored but depending on the answer to that question, this answer may need to be adjusted to pass references rather than using merge field syntax (which only works within the VF page itself).
If you need to get the generated unique id for the node on the page, you should use the {!$Component.yourTagID}
merge field syntax to do so. This mechanism will give you the full id attribute of the element in the page as it was rendered in the browser.
If the path to the component cannot be found as you wrote it, your path is likely incorrect or ambiguous in the page context. It will not throw an error when this happens, it will just return an empty string. i.e. var myVar = '';
where you expected var myVar = 'j_id0:j_id2:myThing';
You can verify what's happening by looking at the source in the browser.
For instance:
(function (window, document) {
var layout = document.getElementById('{!$Component.layout}');
var menu = document.getElementById('{!$Component.menu}');
var menuLink = document.getElementById('{!$Component.menuLink}');
or something more targeted to a specific path from the <apex:page id="myPage">
to the element itself:
(function (window, document) {
var layout = document.getElementById('{!$Component.myPage.myForm.myPageBlock.layout}');
var menu = document.getElementById('{!$Component.myPage.myForm.myPageBlock.menu}');
var menuLink = document.getElementById('{!$Component.myPage.myForm.myPageBlock.menuLink}');
or if you want to pass the DOM id as a function param:
(function (window, document, layoutId, menuId, menuLinkId) {
var layout = document.getElementById(layoutId);
var menu = document.getElementById(menuId);
var menuLink = document.getElementById(menuLinkId);
// check for nulls
// content trimmed for brevity
}(this, this.document, '{!$Component.layout}', '{!$Component.menu}', '{!$Component.menuLink}'));
In this case, it was a simple typo: the ID value is cAsE-sEnSiTiVe.
alert('called' + document.getElementById("thepage:theform:atextId").value);
// ^ This P was capitalized
Always make sure you're matching the case for your Id values.
As a more practical solution, consider using $Component:
alert('called' + document.getElementById("{!$Component.thePage.theform.atextId}").value);
Using $Component will resolve the element Id correctly (in a case-insensitive manner) so you don't have to worry about it as much.
Best Answer
You might have better luck using the
ends with
selector:Or better yet, use style classes:
As for why what you have doesn't work, you have to reference all the parent tags as well. If they don't have
id
attributes specified, it's more of a pain and not worth the trouble in my opinion, hence the above recommendations. Have a look at theVisualforce Developer Guide
(emphasis mine):See also: