출처
http://noritersand.tistory.com/218
관련 문서
- http://api.jquery.com/on/
- http://www.websamo.com/bbs/board.php?bo_table=lecture&wr_id=161&sca=%EC%A0%9C%EC%9D%B4%EC%BF%BC%EB%A6%AC&sfl=wr_subject&stx=on&sop=and
.on()
하나 혹은 그 이상의 요소(노드)에 이벤트 핸들러를 부착(?)한다. (Attach an event handler function for one or more events to the selected elements)
.on( events [, selector] [, data], handler( eventObject ) )
- events: 공백으로 구분된 하나 이상의 이벤트 타입과 옵션인 네임스페이스. "click", "keydown.myPlugin", ".myPlugin" 등이 있음
- selector: 이벤트가 발생할 요소들의 자손을 찾는 선택자. 선택자가 null 이거나 생략됐다면 이벤트는 선택된 요소에 한해서 반응한다.
- data: 이벤트가 발생할 때 핸들러에 전달할 데이터
- handler(eventObject): 이벤트가 발생되면 실행될 기능. false를 반환하는 함수라면 간단하게 false 를 직접 인자로 하면 된다.
.on( events-map [, selector] [, data] )
- events-map: 공백으로 구분된 하나 이상의 이벤트 타입과 선택적인 네임스페이스로 구성된 키(keys)들과 값(values)들의 문자열.
on()은 jQuery 1.7부터 bind(), delegate(), live()를 대체하는 메서드로 이벤트 핸들러 바인딩에 필요한 기존 메서드의 모든 기능을 제공한다. jQuery는 앞의 세 메서드 대신 on()을 사용할 것을 권장하고 있다.
$(선택자).on(이벤트타입 [, 자손선택자] [, 데이터], 핸들러())
on()은 기본적으로 jQuery 객체를 리턴하지만 메서드의 전달인자로 추가적인 선택자를 명시하면 최초 요소의 하위를 재탐색하여 리턴한다.
$("#btn_close").on("click", function() {
self.close();
});
$("#btn").bind("click", function() {});
$("#btn").on("click", function() {});
$("body").delegate("#btn", "click", function() {});
$("body").on("click", "#btn", function() {});
$("#btn").live("click", function() {});
$("#btn").on("click", function() {});
사용방법은 bind(), delegate(), live()와 대동소이하다.
주의해야 할 것은 on을 사용한다고 해서 무조건 동적처리가 가능한것은 아니라는 점. 여기서 동적 처리란 HTML 문서의 로딩이 완료된 후 스크립트에 의해 HTML 구성요소가 추가로 생성되었을 때, 이 구성요소에서 발생하는 이벤트를 감지하여 핸들러를 호출하는 과정을 의미한다.
<script>
jQuery(document).ready(function() {
$("#demo6 li").on("click", function() {
alert($(this).text());
});
$("#demo6 button").on("click", function() {
$("#demo6 ul").append("<li>added</li>");
});
});
</script>
<div id="demo6">
<button>ADD</button>
<ul>
<li>test1</li>
<li>test2</li>
<li>test3</li>
</ul>
</div>
- test1
- test2
- test3
위의 ADD와 test#버튼을 눌러볼 것. ADD에 의해 추가된 요소는 클릭 이벤트가 발생해도 아무런 반응이 없다.
반면 다음의 경우:
<script>
jQuery(document).ready(function() {
$("#demo7").on("click", "li", function() {
alert($(this).text());
});
$("#demo7 button").on("click", function() {
$("#demo7 ul").append("<li>added</li>");
});
});
</script>
- test1
- test2
- test3
ADD와 test#를 눌러보면 ADD에 의해 추가된 요소도 클릭 이벤트에 반응하는 것을 확인할 수 있다.
$(선택자).on(이벤트타입 [, 추가선택자] [, 데이터], 핸들러());
1. 파라미터에 추가선택자가 없다 → bind()와 동일, 동적처리 불가능
2. 파라미터에 추가선택자가 있다 → delegate()와 동일, 동적처리 가능
추가선택자를 명시하면 하위 요소에서 발생한 이벤트에도 반응하도록 핸들러를 등록한다. 이벤트가 발생한 노드가 추가선택자와 일치하면 핸들러가 작동하는 방식이다. 이벤트 버블링*을 이용한 방식으로 순수 자바스크립트에서도 구현할 수 있지만 jQuery에선 버블링을 지원하지 않는 브라우저를 위한 호환처리가 되어있다. (1.x 기준)
* 버블링이란?
계층적 구조의 포함되어 있는 특정 엘리먼트에 이벤트가 발생할 경우, 연쇄적인 반응이 일어나는데 이를 이벤트 전파라고 한다. 이벤트 대상 엘리먼트를 시작으로 최하위 엘리먼트까지 이벤트가 전파가 되는 것을 캡쳐링(Capturing)이라고 하고, 반대로 최상위 엘리먼트에서 대상 엘리먼트까지 이벤트가 전파되는 것을 버블링(Bubbling)이라 한다.
events-map
on() 메서드의 파라미터인 events-map은 공백으로 구분되는 문자열, 혹은 JSON, map이 올 수 있다. 여기서 map은 생략한다.
a. 동적 처리 불가능한 방식 - 공백으로 구분된 문자열
'공백으로 구분되는 문자열'이란:
"eventType1 eventType2 eventType3..."
실제로 사용해보자:
<script>
jQuery(document).ready(function() {
$("#demo8 button").on("click mouseout", function() {
alert("이벤트 발생!");
});
});
</script>
<div id="demo8">
<button>눌러보세요</button>
</div>
b. 동적 처리 불가능한 방식 - JSON
반면, JSON의 경우에는 각기 다른 핸들러를 전달할 수 있다:
<script>
jQuery(document).ready(function() {
$("#demo9 button").on({
click: function() {
$("#demo9 #result").html("<span style='color: red;'>click event</span>")
},
mouseover: function() {
$("#demo9 #result").html("<span style='color: blue;'>mouseover event</span>")
}
});
});
</script>
<div id="demo9">
<button>눌러보세요</button>
<div id="result"></div>
</div>
c. 동적 처리 가능한 방식(이벤트 버블) - 공백으로 구분된 문자열
<script>
jQuery(document).ready(function() {
var cnt = 0;
$("#demo10").on("click mouseover", "div", function() {
$("#demo10 #cnt").text(cnt);
cnt++;
});
$("#demo10 button").on("click", function() {
$("#demo10").append("<div>추가된 놈</div>");
});
});
</script>
<div id="demo10">
<button>ADD</button> <span id="cnt"></span>
<div>원래 있던 놈</div>
</div>
d. 동적 처리 가능한 방식(이벤트 버블) - JSON
<script>
jQuery(document).ready(function() {
$("#demo11").on({
click: function() {
$("#demo11 #result").html("<span style='color: red;'>click event</span>")
},
mouseover: function() {
$("#demo11 #result").html("<span style='color: blue;'>mouseover event</span>")
}
}, "div");
$("#demo11 button").on("click", function() {
$("#demo11").append("<div>추가된 놈</div>");
});
});
</script>
<div id="demo11">
<button>ADD</button> <span id="result"></span>
<div>원래 있던 놈</div>
</div>
event namespaces
네임스페이스를 활용하면 하나의 이벤트에 다수의 핸들러를 할당하고 이를 컨트롤 해야 할 때 접근을 쉽게 할 수 있다. 예를 들어 button태그의 click 이벤트 발생 시 한번에 네 개의 함수가 작동하도록 만든다고 하자. 이를 네임스페이스로 바인딩했다면 차후 동적처리로 특정함수를 제거해야 한다고 할 때 해당 네임스페이스를 이벤트타입으로 지정하여 삭제할 수 있다.
다음은 하나의 클릭 이벤트에 다수의 핸들러를 네임스페이스를 이용해 바인딩하고 그 중 일부만 삭제하는 예시다:
<script>
jQuery(document).ready(function() {
$("#demo12 #do").on("click.first", function() {
alert("first action");
});
$("#demo12 #do").on({
"click.second": function() {
alert("second action");
},
"click.third": function() {
alert("third action");
}
});
});
function fn_del() {
$("#demo12 button").off("click.second");
}
</script>
<div id="demo12">
<button id="do">ANG?</button>
<button onclick="fn_del()">second action 삭제</button>
</div>
위에서 "click.first"는 click 이벤트의 first 네임스페이스를 의미하는데 여기서 네임스페이스는 1depth 이상일 수 있다. 이 말은 "click.action.event.first"라는 네임스페이스도 가능하다는 뜻이다.
data
on() 메서드의 인자로 null 이나 undefined 가 아닌 값이 지정되면 이 값은 event.data 속성을 통해 핸들러에 전달된다. data 인자는 어떤 타입이라도 사용할 수 있지만 만일 문자열을 사용했다면 데이터 선택 실수를 방지하기 위해 selector를 반드시 제공하거나 명시적으로 null을 지정해야 한다.
<script>
jQuery(document).ready(function() {
function foo(event) {
alert(event.data.txt1 + event.data.txt2);
}
$("#demo13 #do").on("click", {txt1: "hello", txt2: " world!"}, foo);
});
</script>
<div id="demo13">
<button id="do">HI</button>
</div>
* 아래는 bind, delegate, live 메서드의 간략 정리
.bind()
.bind( eventType [, eventData] , handler( eventObject ) )
- eventType: 공백으로 구분된 하나 이상의 JavaScript 이벤트를 포함하는 문자열, "click", "keydown" 또는 사용자 정의 함수명
bind()는 jQuery로 선택한 DOM 객체에 이벤트 핸들러를 등록하는 메서드다.
<script>
jQuery(document).ready(function() {
$("#demo1 li").bind("click", function() {
alert($(this).text());
});
});
</script>
<div id="demo1">
<ul>
<li>test1</li>
<li>test2</li>
<li>test3</li>
</ul>
</div>
- test1
- test2
- test3
bind()는 이벤트 핸들링의 기본이라고 할 수 있는데, jQuery는 매우 빈번하게 일어나는 이벤트에 대해선 bind()를 사용하지 않고 사용할 수 있도록 일종의 전용 바인더를 제공하는데 jQuery.click(), jQuery.hover() 등이 그것이다. 그런데 문제는 이런 bind 계열 메서드가 동적으로 생성된 노드에 접근하도록 하는게 번거롭다는 점이다.
다음 예를 보면:
<script>
jQuery(document).ready(function() {
$("#demo2 button").bind("click", function() {
$("#demo2 ul").append("<li>added</li>");
});
$("#demo2 li").bind("click", function() {
alert($(this).text());
});
});
</script>
<div id="demo2">
<button>add</button>
<ul>
<li>test1</li>
<li>test2</li>
<li>test3</li>
</ul>
</div>
- test1
- test2
- test3
add 버튼 클릭 시 동적으로 생성된 노드의 li 요소는 클릭해도 아무런 반응을 보이지 않는다. 이는 bind()에 전달된 jQuery 객체가 document.ready 시점에 생성된 노드로 한정되기 때문이다. 따라서 문서로딩 이후 추가되는 노드에 이벤트를 등록하고자 한다면 다음처럼 소스를 고쳐야 한다:
<script>
jQuery(document).ready(function() {
$("#demo3 button").bind("click", function() {
$("#demo3 li").unbind("click");
$("#demo3 ul").append("<li>added</li>").find($("li")).click(function() {
alert($(this).text());
});
});
$("#demo3 li").bind("click", function() {
alert($(this).text());
});
});
</script>
- test1
- test2
- test3
이 외에는 버블링을 이용하는 방법이 있다.
<script>
jQuery(document).ready(function() {
$("#demo3 button").bind("click", function() {
$("#demo3 ul").append("<li>added</li>");
});
$("#demo3").bind("click", function(e) {
if (e.target.tagName == "LI") {
alert($(e.target).text());
}
})
});
</script>
.delegate()
.delegate( selector, eventType, handler )
- eventType: 공백으로 구분된 하나 이상의 JavaScript 이벤트를 포함하는 문자열, "click", "keydown" 또는 사용자 정의 함수명
다음은 delegate()를 사용해 동적으로 생성된 노드를 핸들링하는 코드다:
<script>
jQuery(document).ready(function() {
$("#demo4 button").bind("click", function() {
$("#demo4 ul").append("<li>added</li>");
});
$("#demo4").delegate("li", "click", function() {
alert($(this).text());
});
});
</script>
- test1
- test2
- test3
.live()
.live( events, handler )
예시:
<script src="http://code.jquery.com/jquery-1.6.4.js"></script>
<script>
jQuery(document).ready(function() {
$("#demo5 button").bind("click", function() {
$("#demo5 ul").append("<li>added</li>");
});
$("#demo5 li").live("click", function() {
alert($(this).text());
});
});
</script>
<div id="demo5">
<button>ADD</button>
<ul>
<li>test1</li>
<li>test2</li>
<li>test3</li>
</ul>
</div>
(데모는 생략)
live() 메서드는 1.7 이후 사용이 완전히 금지(실제로 1.7.x 이상의 라이브러리에서는 없는 메서드라며 에러가 난다)되었기 때문에 부득이하게 1.6.4 라이브러리를 이용하여 작성했다 bind(), delegate()와 대동소이하다.