끔찍한 이벤트 리스너를 파괴하지 않고 innerHTML에 추가 할 수 있습니까?
다음 예제 코드에서는 onclick
"foo"라는 텍스트가 포함 된 범위에 이벤트를 연결합니다 . 핸들러는 alert()
.
그러나 부모 노드에 할당하면 innerHTML
이 onclick
이벤트가 파괴됩니다. "foo"를 클릭하면 경고 상자가 표시되지 않습니다.
이 문제를 해결할 수 있습니까?
<html>
<head>
<script type="text/javascript">
function start () {
myspan = document.getElementById("myspan");
myspan.onclick = function() { alert ("hi"); };
mydiv = document.getElementById("mydiv");
mydiv.innerHTML += "bar";
}
</script>
</head>
<body onload="start()">
<div id="mydiv" style="border: solid red 2px">
<span id="myspan">foo</span>
</div>
</body>
</html>
안타깝게도에 할당 innerHTML
하면 추가하려는 경우에도 모든 하위 요소가 파괴됩니다. 마이너 노드 (및 해당 이벤트 처리기)를 보존 비용 DOM 함수 를 사용 합니다 .
function start() {
var myspan = document.getElementById("myspan");
myspan.onclick = function() { alert ("hi"); };
var mydiv = document.getElementById("mydiv");
mydiv.appendChild(document.createTextNode("bar"));
}
편집 : 의견에서 Bob의 솔루션. 답을 게시하세요, 밥! 대한 신용을 얻으십시오. :-)
function start() {
var myspan = document.getElementById("myspan");
myspan.onclick = function() { alert ("hi"); };
var mydiv = document.getElementById("mydiv");
var newcontent = document.createElement('div');
newcontent.innerHTML = "bar";
while (newcontent.firstChild) {
mydiv.appendChild(newcontent.firstChild);
}
}
사용하면 .insertAdjacentHTML()
이벤트 리스너가 유지 되며 모든 주요 브라우저에서 지원 됩니다. .NET의 간단한 한 줄 대체입니다 .innerHTML
.
var html_to_insert = "<p>New paragraph</p>";
// with .innerHTML, destroys event listeners
document.getElementById('mydiv').innerHTML += html_to_insert;
// with .insertAdjacentHTML, preserves event listeners
document.getElementById('mydiv').insertAdjacentHTML('beforeend', html_to_insert);
'beforeend'
요소에 인수 지정은 HTML 콘텐츠를 삽입합니다. 옵션은 'beforebegin'
, 'afterbegin'
, 'beforeend'
,와 'afterend'
. 해당 위치는 다음과 가변됩니다.
<!-- beforebegin -->
<div id="mydiv">
<!-- afterbegin -->
<p>Existing content in #mydiv</p>
<!-- beforeend -->
</div>
<!-- afterend -->
이제 2012 년이되었습니다. jQuery는 정확히 수행하는 추가 및 추가 기능을 가지고 있으며 현재 콘텐츠에 영향을주지 않고 콘텐츠를 추가합니다. 매우 유용하다.
약간의 (하지만 관련이있는) 제쳐두고, jquery (v1.3)와 같은 자바 펼쳐보기 라이브러리를 사용하여 dom 조작을 수행하는 경우 라이브 이벤트를 사용하여 다음과 같은 접근을 수 있습니다.
$("#myspan").live("click", function(){
alert('hi');
});
모든 종류의 jquery 조작 중에 항상 해당 선택기에 적용됩니다. 라이브 이벤트를 참조하십시오 : docs.jquery.com/events/live JQuery와 조작은 다음을 참조하십시오 docs.jquery.com/manipulation를
something.innerHTML + = '원하는대로 추가';
그것은 나를 위해 일했습니다. 이 솔루션을 사용하여 입력 텍스트에 버튼을 추가했습니다.
코드가 적고 멋진 dom 항목으로 작업하는 것보다 읽기가 쉽기 때문에 문자열로 삽입 할 마크 업을 만들었습니다.
그런 다음 해당 요소의 유일한 자식을 가져와 본문에 연결할 수 있도록 임시 요소의 innerHTML을 만들었습니다.
var html = '<div>';
html += 'Hello div!';
html += '</div>';
var tempElement = document.createElement('div');
tempElement.innerHTML = html;
document.getElementsByTagName('body')[0].appendChild(tempElement.firstChild);
또 다른 대안이 있습니다 setAttribute
. 이벤트 리스너를 추가하는 대신 사용 하는 것입니다. 이렇게 :
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Demo innerHTML and event listeners</title>
<style>
div {
border: 1px solid black;
padding: 10px;
}
</style>
</head>
<body>
<div>
<span>Click here.</span>
</div>
<script>
document.querySelector('span').setAttribute("onclick","alert('Hi.')");
document.querySelector('div').innerHTML += ' Added text.';
</script>
</body>
</html>
이벤트 핸들러를 잃는 것은 IMO, Javascript가 DOM을 처리하는 방식의 버그입니다. 이 동작을 방지하려면 다음을 추가 할 수 있습니다.
function start () {
myspan = document.getElementById("myspan");
myspan.onclick = function() { alert ("hi"); };
mydiv = document.getElementById("mydiv");
clickHandler = mydiv.onclick; // add
mydiv.innerHTML += "bar";
mydiv.onclick = clickHandler; // add
}
다음과 같이 할 수 있습니다.
var anchors = document.getElementsByTagName('a');
var index_a = 0;
var uls = document.getElementsByTagName('UL');
window.onload=function() {alert(anchors.length);};
for(var i=0 ; i<uls.length; i++)
{
lis = uls[i].getElementsByTagName('LI');
for(var j=0 ;j<lis.length;j++)
{
var first = lis[j].innerHTML;
string = "<img src=\"http://g.etfv.co/" + anchors[index_a++] +
"\" width=\"32\"
height=\"32\" /> " + first;
lis[j].innerHTML = string;
}
}
가장 쉬운 방법은 배열을 사용하고 여기에 요소를 푸시 한 다음 배열 후속 값을 배열에 동적으로 삽입하는 것입니다. 내 코드는 다음과 같습니다.
var namesArray = [];
function myclick(){
var readhere = prompt ("Insert value");
namesArray.push(readhere);
document.getElementById('demo').innerHTML= namesArray;
}
예, angular / vue / react / etc 프레임 워크와 유사한이 접근 방식과 유사한 onclick="sayHi()"
템플릿에서 직접 인수를 사용하여 이벤트를 바인딩하면 가능합니다 <body onload="start()">
. 여기<template>
와 같이 '동적'html에서 작업 하는 데 사용할 수도 있습니다 . 엄격한 눈에 거슬리지 않는 js는 아니지만 소규모 프로젝트 에는 허용 됩니다.
function start() {
mydiv.innerHTML += "bar";
}
function sayHi() {
alert("hi");
}
<body onload="start()">
<div id="mydiv" style="border: solid red 2px">
<span id="myspan" onclick="sayHi()">foo</span>
</div>
</body>
헤더와 데이터가있는 모든 객체 배열의 경우.jsfiddle
https://jsfiddle.net/AmrendraKumar/9ac75Lg0/2/
<table id="myTable" border='1|1'></table>
<script>
const userObjectArray = [{
name: "Ajay",
age: 27,
height: 5.10,
address: "Bangalore"
}, {
name: "Vijay",
age: 24,
height: 5.10,
address: "Bangalore"
}, {
name: "Dinesh",
age: 27,
height: 5.10,
address: "Bangalore"
}];
const headers = Object.keys(userObjectArray[0]);
var tr1 = document.createElement('tr');
var htmlHeaderStr = '';
for (let i = 0; i < headers.length; i++) {
htmlHeaderStr += "<th>" + headers[i] + "</th>"
}
tr1.innerHTML = htmlHeaderStr;
document.getElementById('myTable').appendChild(tr1);
for (var j = 0; j < userObjectArray.length; j++) {
var tr = document.createElement('tr');
var htmlDataString = '';
for (var k = 0; k < headers.length; k++) {
htmlDataString += "<td>" + userObjectArray[j][headers[k]] + "</td>"
}
tr.innerHTML = htmlDataString;
document.getElementById('myTable').appendChild(tr);
}
</script>
나는 게으른 프로그래머입니다. 추가 타이핑처럼 보이기 때문에 DOM을 사용하지 않습니다. 나에게는 코드가 적을수록 좋습니다. "foo"를 대체하지 않고 "bar"를 추가하는 방법은 다음과 같습니다.
function start(){
var innermyspan = document.getElementById("myspan").innerHTML;
document.getElementById("myspan").innerHTML=innermyspan+"bar";
}
'IT' 카테고리의 다른 글
python NameError : 전역 이름 '__file__'이 정의되지 않습니다. (0) | 2020.09.03 |
---|---|
EF 6 및 Code First 마이그레이션의 동일한 DB 및 애플리케이션에있는 여러 DB 다수 (0) | 2020.09.03 |
C ++ catch 블록-값 또는 참조로 발생합니까? (0) | 2020.09.03 |
node.js postgresql 모듈을 사용하는 적절한 방법은 무엇입니까? (0) | 2020.09.03 |
벡터를 0으로 초기화 C ++ / C ++ 11 (0) | 2020.09.03 |