728x90
반응형
커스텀 태그 안에 커스텀 태그가 위치해 있을 경우 서로간에 필요한 속성이라던가 내용 등을 읽어올 필요가 있다. 이럴 경우에는 다음과 같은 방법으로 접근할 수 있다.
getParent 메소드를 반복해서 호출해서 depth가 몇단계인지 알아내는 예제이다.
SimpleTag 인터넾이스와 Tag인터페이스는 JspTag 인터페이스를 구현한 것이다. SimpleTag의 getParent 메소드는 JspTag를 리턴하는데 반해 Tag(클래식 커스텀 태그)의 getParent 메소드는 Tag를 리턴한다.
이 점은 중요한데 왜냐하면 이 차이로 인해 Tag가 SimpleTag의 부모 태그가 될 수 있지만, SimpleTag가 Tag의 부모 태그가 될 수 없기 때문이다.
아래 내용은 Tag가 SimpleTag의 부모 태그로서 사용 되어질 경우 부모 태그의 attribute를 알아오는 예제이다.
tld 파일에서는 ClassicParent 부모 태그와 SimpleInner 자식 태그를 정의 한다. ClassicParent 에는 name 이란 attribute가 존재 한다.
ClassicParent와 SimpleInner태그를 구현한 각 Java 파일은 위 소스와 같이 구현한다.
SimpleInner.java 파일에서 눈여겨 볼 것은 getParent 메소드를 통해 리턴 받은 JspTag형 객체를 ClassicParent로 형변환 해서 사용하는 것이다.
ClassicParent.java 파일에는 setter와 getter가 있는데, 자동으로 setter는 tld 에서 정의한 name이란 attribute에 값을 할당해 주며, getter는 SimpleInner.java에서 사용한다.
물론 ClassicParent.java의 doStartTag()에서 EVAL_BODY_INCLUDE를 리턴해 주어야지만 자식 태그인 SimpleInner가 실행 되게 된다.
중요한 것은 자식 태그에서 부모 태그의 정보를 읽을 수는 있어도 부모에서 자식 태그의 정보를 읽을 수는 없다.
따라서 부모에서 자식 태그를 알기 위해서는 반대로 자식에수 부모 쪽에 값을 셋팅 하는 방법을 사용해야 한다.
다음 예제를 통해서 부모가 자식의 속성 값을 알아내는 방법을 살펴보자.
menuTag.tld 파일에선 Menu태그와 MenuItem 태그를 정의한다. MenuItem 태그가 child 태그이며, itemValue라는 attribute를 가지고 있다.
Menu 태그를 정의한 Menu.java 파일에서는 ArrayList 변수 addMenuItem 메소드를 하나 만들고 값을 받아 하나씩 추가할 수 있도록 한다. addMenuItem 메소드는 child 태그인 MenuItem 클래스에서 사용할 메소드이다.
MenuItem 태그를 정의한 MenuItem.java에서는 doEndTag 부분에서 parent 객체를 하나 생성한 후 Menu 클래스에 있는 addMenuItem 메소드를 호출해 자신이 가지고 있는 itemValue라는 속성을 넘겨준다.
마지막으로 Menu 클래스의 doEndTag 메소드에서는 child 태그로부터 넘겨 받게 된 값을 화면에 보여준다.
myCustomTag3.tld
<?xml version="1.0" encoding="UTF-8"?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd" version="2.0">
<tlib-version>0.1</tlib-version>
<uri>Nested</uri>
<tag>
<description>NestedLevel Check</description>
<name>NestedLevel</name>
<tag-class>com.example.tag.NestedLevelTag</tag-class>
<body-content>JSP</body-content>
</tag>
</taglib>
<?xml version="1.0" encoding="UTF-8"?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd" version="2.0">
<tlib-version>0.1</tlib-version>
<uri>Nested</uri>
<tag>
<description>NestedLevel Check</description>
<name>NestedLevel</name>
<tag-class>com.example.tag.NestedLevelTag</tag-class>
<body-content>JSP</body-content>
</tag>
</taglib>
NestedLevelTag.java
package com.example.tag;
import java.io.IOException;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;
public class NestedLevelTag extends TagSupport {
private int nestLevel = 0;
public int doStartTag() throws JspException {
nestLevel = 0;
Tag parent = getParent();
while(parent != null) {
parent = parent.getParent();
nestLevel++;
}
try {
pageContext.getOut().println("<br>Tag nested level : " + nestLevel);
} catch (IOException e) {
// TODO: handle exception
throw new JspException("IOException - " + e.toString());
}
return EVAL_BODY_INCLUDE;
}
}
package com.example.tag;
import java.io.IOException;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;
public class NestedLevelTag extends TagSupport {
private int nestLevel = 0;
public int doStartTag() throws JspException {
nestLevel = 0;
Tag parent = getParent();
while(parent != null) {
parent = parent.getParent();
nestLevel++;
}
try {
pageContext.getOut().println("<br>Tag nested level : " + nestLevel);
} catch (IOException e) {
// TODO: handle exception
throw new JspException("IOException - " + e.toString());
}
return EVAL_BODY_INCLUDE;
}
}
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="mine" uri="Nested" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<mine:NestedLevel>
<mine:NestedLevel>
<mine:NestedLevel/>
</mine:NestedLevel>
</mine:NestedLevel>
</body>
</html>
pageEncoding="UTF-8"%>
<%@ taglib prefix="mine" uri="Nested" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<mine:NestedLevel>
<mine:NestedLevel>
<mine:NestedLevel/>
</mine:NestedLevel>
</mine:NestedLevel>
</body>
</html>
결과물
Insert title here Tag nested level : 0
Tag nested level : 1
Tag nested level : 2
Tag nested level : 1
Tag nested level : 2
getParent 메소드를 반복해서 호출해서 depth가 몇단계인지 알아내는 예제이다.
SimpleTag 인터넾이스와 Tag인터페이스는 JspTag 인터페이스를 구현한 것이다. SimpleTag의 getParent 메소드는 JspTag를 리턴하는데 반해 Tag(클래식 커스텀 태그)의 getParent 메소드는 Tag를 리턴한다.
이 점은 중요한데 왜냐하면 이 차이로 인해 Tag가 SimpleTag의 부모 태그가 될 수 있지만, SimpleTag가 Tag의 부모 태그가 될 수 없기 때문이다.
아래 내용은 Tag가 SimpleTag의 부모 태그로서 사용 되어질 경우 부모 태그의 attribute를 알아오는 예제이다.
classicParent.tld
<?xml version="1.0" encoding="UTF-8"?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd" version="2.0">
<tlib-version>0.1</tlib-version>
<uri>ClassicParentTag</uri>
<tag>
<description>classic parent</description>
<name>ClassicParent</name>
<tag-class>com.example.tag.MyClassicParent</tag-class>
<body-content>JSP</body-content>
<attribute>
<name>name</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
<tag>
<description>Simple Inner Tag</description>
<name>SimpleInner</name>
<tag-class>com.example.tag.SimpleInner</tag-class>
<body-content>empty</body-content>
</tag>
</taglib>
<?xml version="1.0" encoding="UTF-8"?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd" version="2.0">
<tlib-version>0.1</tlib-version>
<uri>ClassicParentTag</uri>
<tag>
<description>classic parent</description>
<name>ClassicParent</name>
<tag-class>com.example.tag.MyClassicParent</tag-class>
<body-content>JSP</body-content>
<attribute>
<name>name</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
<tag>
<description>Simple Inner Tag</description>
<name>SimpleInner</name>
<tag-class>com.example.tag.SimpleInner</tag-class>
<body-content>empty</body-content>
</tag>
</taglib>
SimpleInner.java
package com.example.tag;
import java.io.IOException;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.SimpleTagSupport;
public class SimpleInner extends SimpleTagSupport {
public void doTag() throws JspException, IOException {
MyClassicParent parent = (MyClassicParent) getParent();
getJspContext().getOut().print("Parent attribute is : " + parent.getName());
}
}
package com.example.tag;
import java.io.IOException;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.SimpleTagSupport;
public class SimpleInner extends SimpleTagSupport {
public void doTag() throws JspException, IOException {
MyClassicParent parent = (MyClassicParent) getParent();
getJspContext().getOut().print("Parent attribute is : " + parent.getName());
}
}
MyClassicParent.java
package com.example.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
public class MyClassicParent extends TagSupport {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int doStartTag() throws JspException {
return EVAL_BODY_INCLUDE;
}
}
package com.example.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
public class MyClassicParent extends TagSupport {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int doStartTag() throws JspException {
return EVAL_BODY_INCLUDE;
}
}
classicParentTag.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="mine" uri="ClassicParentTag" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<mine:ClassicParent name="aaa">
<mine:SimpleInner/>
</mine:ClassicParent>
</body>
</html>
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="mine" uri="ClassicParentTag" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<mine:ClassicParent name="aaa">
<mine:SimpleInner/>
</mine:ClassicParent>
</body>
</html>
결과물
Insert title here Parent attribute is : aaa
tld 파일에서는 ClassicParent 부모 태그와 SimpleInner 자식 태그를 정의 한다. ClassicParent 에는 name 이란 attribute가 존재 한다.
ClassicParent와 SimpleInner태그를 구현한 각 Java 파일은 위 소스와 같이 구현한다.
SimpleInner.java 파일에서 눈여겨 볼 것은 getParent 메소드를 통해 리턴 받은 JspTag형 객체를 ClassicParent로 형변환 해서 사용하는 것이다.
ClassicParent.java 파일에는 setter와 getter가 있는데, 자동으로 setter는 tld 에서 정의한 name이란 attribute에 값을 할당해 주며, getter는 SimpleInner.java에서 사용한다.
물론 ClassicParent.java의 doStartTag()에서 EVAL_BODY_INCLUDE를 리턴해 주어야지만 자식 태그인 SimpleInner가 실행 되게 된다.
중요한 것은 자식 태그에서 부모 태그의 정보를 읽을 수는 있어도 부모에서 자식 태그의 정보를 읽을 수는 없다.
따라서 부모에서 자식 태그를 알기 위해서는 반대로 자식에수 부모 쪽에 값을 셋팅 하는 방법을 사용해야 한다.
다음 예제를 통해서 부모가 자식의 속성 값을 알아내는 방법을 살펴보자.
menuTag.tld
<?xml version="1.0" encoding="UTF-8"?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd" version="2.0">
<tlib-version>0.1</tlib-version>
<uri>MenuTag</uri>
<tag>
<description>parent menu tag</description>
<name>Menu</name>
<tag-class>com.example.tag.Menu</tag-class>
<body-content>JSP</body-content>
</tag>
<tag>
<description>child menu tag</description>
<name>MenuItem</name>
<tag-class>com.example.tag.MenuItem</tag-class>
<body-content>JSP</body-content>
<attribute>
<name>itemValue</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
</taglib>
<?xml version="1.0" encoding="UTF-8"?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd" version="2.0">
<tlib-version>0.1</tlib-version>
<uri>MenuTag</uri>
<tag>
<description>parent menu tag</description>
<name>Menu</name>
<tag-class>com.example.tag.Menu</tag-class>
<body-content>JSP</body-content>
</tag>
<tag>
<description>child menu tag</description>
<name>MenuItem</name>
<tag-class>com.example.tag.MenuItem</tag-class>
<body-content>JSP</body-content>
<attribute>
<name>itemValue</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
</taglib>
Menu.java
package com.example.tag;
import java.util.ArrayList;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
public class Menu extends TagSupport {
private ArrayList<String> items;
public void addMenuItem(String item) {
items.add(item);
}
public int doStartTag() throws JspException {
items = new ArrayList<String>();
return EVAL_BODY_INCLUDE;
}
public int doEndTag() throws JspException {
try {
pageContext.getOut().println("Menu items are : " + items);
} catch (Exception e) {
// TODO: handle exception
throw new JspException("Exception : " + e.toString());
}
return EVAL_PAGE;
}
}
package com.example.tag;
import java.util.ArrayList;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
public class Menu extends TagSupport {
private ArrayList<String> items;
public void addMenuItem(String item) {
items.add(item);
}
public int doStartTag() throws JspException {
items = new ArrayList<String>();
return EVAL_BODY_INCLUDE;
}
public int doEndTag() throws JspException {
try {
pageContext.getOut().println("Menu items are : " + items);
} catch (Exception e) {
// TODO: handle exception
throw new JspException("Exception : " + e.toString());
}
return EVAL_PAGE;
}
}
MenuItem.java
package com.example.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
public class MenuItem extends TagSupport {
private String itemValue;
public void setItemValue(String itemValue) {
this.itemValue = itemValue;
}
public int doStartTag() throws JspException {
return EVAL_BODY_INCLUDE;
}
public int doEndTag() throws JspException {
Menu parent = (Menu) getParent();
parent.addMenuItem(itemValue);
return EVAL_PAGE;
}
}
package com.example.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
public class MenuItem extends TagSupport {
private String itemValue;
public void setItemValue(String itemValue) {
this.itemValue = itemValue;
}
public int doStartTag() throws JspException {
return EVAL_BODY_INCLUDE;
}
public int doEndTag() throws JspException {
Menu parent = (Menu) getParent();
parent.addMenuItem(itemValue);
return EVAL_PAGE;
}
}
menuItem.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="mine" uri="MenuTag" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<mine:Menu>
<mine:MenuItem itemValue="Dogs"></mine:MenuItem>
<mine:MenuItem itemValue="Cats"></mine:MenuItem>
<mine:MenuItem itemValue="Horses"></mine:MenuItem>
</mine:Menu>
</body>
</html>
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="mine" uri="MenuTag" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<mine:Menu>
<mine:MenuItem itemValue="Dogs"></mine:MenuItem>
<mine:MenuItem itemValue="Cats"></mine:MenuItem>
<mine:MenuItem itemValue="Horses"></mine:MenuItem>
</mine:Menu>
</body>
</html>
결과물
Insert title here Menu items are : [Dogs, Cats, Horses]
menuTag.tld 파일에선 Menu태그와 MenuItem 태그를 정의한다. MenuItem 태그가 child 태그이며, itemValue라는 attribute를 가지고 있다.
Menu 태그를 정의한 Menu.java 파일에서는 ArrayList 변수 addMenuItem 메소드를 하나 만들고 값을 받아 하나씩 추가할 수 있도록 한다. addMenuItem 메소드는 child 태그인 MenuItem 클래스에서 사용할 메소드이다.
MenuItem 태그를 정의한 MenuItem.java에서는 doEndTag 부분에서 parent 객체를 하나 생성한 후 Menu 클래스에 있는 addMenuItem 메소드를 호출해 자신이 가지고 있는 itemValue라는 속성을 넘겨준다.
마지막으로 Menu 클래스의 doEndTag 메소드에서는 child 태그로부터 넘겨 받게 된 값을 화면에 보여준다.
728x90
반응형
'Java > Servlet & JSP' 카테고리의 다른 글
웹 애플리케이션 보안 (4) | 2009.04.10 |
---|---|
웹 애플리케이션 배포하기 (0) | 2009.04.07 |
클래식 커스텀 태그 (0) | 2009.03.26 |
사용자 정의 태그 개발 (1) (0) | 2009.03.23 |
커스텀 태그 사용하기 (0) | 2009.03.17 |