вложенные переменные el для тегов jsp - PullRequest
0 голосов
/ 01 июня 2011

Я пытаюсь создать переменную el, которая доступна только внутри тега, чтобы это было что-то вроде:

<mytaglib:mytag>
  Foo is ${foo}
</mytaglib:mytag>

Я следовал документации, которую я могу найти для создания вложенной переменной el, однако $ {foo} доступен после окончания mytag.

Сокращенная версия .tld файла:

<?xml version="1.0" encoding="UTF-8" ?>

<taglib version="2.0" xmlns="http://java.sun.com/xml/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd">
  <description>My Tags</description>
  <tlib-version>1.0</tlib-version>
  <jsp-version>2.0</jsp-version>
  <short-name>mytaglib</short-name>
  <uri>http://www.example.com/taglibs/mytaglib</uri>

  <tag>
    <name>mytag</name>
    <tag-class>com.example.tags.MyTag</tag-class>
    <body-content>JSP</body-content>

    <variable>
      <name-given>used</name-given>
      <variable-class>java.lang.String</variable-class>
      <scope>NESTED</scope>
    </variable>
  </tag>
</taglib>

затем MyTag делает следующее (я удалил весь код, который не должен быть связан):

public class MyTag extends BodyTagSupport
{
  // these are subtags
  private JspFragment below;
  private JspFragment at;

  @Override
  public int doStartTag() throws JspException
  {
    return EVAL_BODY_BUFFERED;
  }

  @Override
  public int doEndTag() throws JspException
  {
    try
    {
      if (aBoolean)
      {
          below.invoke(pageContext.getOut());
      }
      else
      {
          at.invoke(pageContext.getOut());
      }
    }
    catch (IOException e)
    {
      throw new JspException(e);
    }

    return EVAL_PAGE;
  }

  @Override
  public void doInitBody() throws JspException
  {
    pageContext.setAttribute("used", "valueOfUsed");
  }

  @Override
  public int doAfterBody() throws JspException
  {
    try
    {
      bodyContent.writeOut(bodyContent.getEnclosingWriter());
    }
    catch (IOException e)
    {
      throw new JspException(e);
    }

    return SKIP_BODY;
  }
}

Насколько я понимаю, pageContext.setAttribute () из doInitBody должен находиться во вложенной области видимости, и вы оцениваете фактическое тело в doAfterBody. Это должно выглядеть так, хотя слово «used» доступно за пределами тега.

Общий фон ... тег будет выглядеть (упрощенная форма)

<mytaglib:mytag>
  Used is ${used}
</mytaglib:mytag>

или

<mytaglib:mytag>
  <mytaglib:at>
    You are at your limit, you have used: ${used}
  </mytaglib:at>
  <mytaglib:below>
    You are below your limit, you have used: ${used}
  </mytaglib:below>
</mytaglib:mytag>

Как я могу получить $ {used} для определения области действия в mytag?

1 Ответ

1 голос
/ 02 июня 2011

Я всегда ссылаюсь на JSTL как на модель для любых тегов, которые я пишу.Поэтому мой подход заключается в том, чтобы посмотреть на исходный код тега, который делает нечто похожее, и скопировать его.прекрасный пример тега.

Класс LoopTagSupport в JSTL реализует интерфейс javax.servlet.jsp.tagext.TryCatchFinally , а в методе doFinally он вызывает pageContext.removeAttribute () вручную.

Цитирование дистрибутива источника JSTL1.1:

/**
 * Removes any attributes that this LoopTagSupport set.
 *
 * <p> These attributes are intended to support scripting variables with
 * NESTED scope, so we don't want to pollute attribute space by leaving
 * them lying around.
 */
public void doFinally() {
    /*
     * Make sure to un-expose variables, restoring them to their
     * prior values, if applicable.
     */
    unExposeVariables();
}

/**
 * Removes page attributes that we have exposed and, if applicable,
 * restores them to their prior values (and scopes).
 */
private void unExposeVariables() {
    // "nested" variables are now simply removed
if (itemId != null)
        pageContext.removeAttribute(itemId, PageContext.PAGE_SCOPE);
if (statusId != null)
    pageContext.removeAttribute(statusId, PageContext.PAGE_SCOPE);
}
...