Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

appendChild doesn't append the same element twice

I have a valuesArray and a transcribedPoems arrays. If a key is present in both of them, I want to add an option to my select element in both the Transcribed and the All subgroups. If not, then I want the option to be available only in the All subrgoup. This is my code:

let valuesArray = [   
    ["1", "a manuscript"], ["2", "another one"]
]

const transcribedPoems =['2']

let optGroupTranscribed = document.createElement("optgroup");
optGroupTranscribed.setAttribute("label", "Transcribed")

let optGroupAll = document.createElement("optgroup");
optGroupAll.setAttribute("label", "All")

let select = document.createElement("select");

valuesArray.forEach(function(element) {
        
        let option = document.createElement("option");
        option.value = element[0];
        option.text = element[1];
        
        if (transcribedPoems.includes(element[0])) {

            optGroupTranscribed.appendChild(option);
            optGroupAll.appendChild(option);
            
        }  else {
        
            optGroupAll.appendChild(option);
        }
        
    });

select.appendChild(optGroupTranscribed);
select.appendChild(optGroupAll);
    
document.getElementById("manifests").appendChild(select);

The above code doesn't append anything to optGroupTranscribed though. I noticed that if I take the optGroupAll.appendChild(option); out from the condition, then I do have the transcribed poems in the transcribed subgroup (and not in the All one). I want the transcribed poems to also be in the All subgroup though. What am I missing?

What I get:

<select>
  <optgroup label="Transcribed">
    <option value="2">another one</option>
  </optgroup>
  <optgroup label="All">    
    <option value="1">a manuscript</option>
  </optgroup>
</select>

Expected result:

<select>
  <optgroup label="Transcribed">
    <option value="2">another one</option>
  </optgroup>
  <optgroup label="All">    
    <option value="1">a manuscript</option>
    <option value="2">another one</option>
  </optgroup>
</select>
like image 325
HBMCS Avatar asked May 31 '26 15:05

HBMCS


1 Answers

There's an interesting subtlety here. When you use createElement, it creates an instance of HTMLElement (for the subclass option). When you appendChild that element to another element, it is placed in that element's DOM tree.

If you go to place that same element in another element's DOM tree, it would make it so that the overall tree (the one that would contain both optGroupTranscribed and optGroupAll) NOT a tree, since there would be one child option shared by both elements.

Instead, you can call option.cloneNode() to make a copy of the element. So, try changing it to:

optGroupTranscribed.appendChild(option.cloneNode());
like image 113
garrettmills Avatar answered Jun 03 '26 04:06

garrettmills