Как отобразить компонент, только если другой компонент не отображается?
У меня есть страница, к которой пользователь может получить доступ через Android, iPhone, BlackBerry или через неизвестный браузер. У меня 4 rich:panel
s, по одному для каждой платформы, а последняя является общей.
Код:
<rich:panel id="dlAndroid" rendered="#{fn:containsIgnoreCase(request.getHeader('User-Agent'), 'Android')}">
...
</rich:panel>
<rich:panel id="dlIphone" rendered="#{fn:containsIgnoreCase(request.getHeader('User-Agent'), 'iPhone')}">
...
</rich:panel>
<rich:panel id="dlBlackberry" rendered="#{fn:containsIgnoreCase(request.getHeader('User-Agent'), 'BlackBerry')}">
...
</rich:panel>
<rich:panel id="dlGeneric" rendered="#{ ---> WHAT TO WRITE HERE? <--- }">
Как я могу сделать последний rich:panel
только если ни один из других не был предоставлен?
3 ответа
Кстати, на ваш вопрос, указанный в названии, можно дать конкретный ответ:
<rich:panel binding="#{panel1}" ...>
...
</rich:panel>
<rich:panel binding="#{panel2}" ...>
...
</rich:panel>
<rich:panel binding="#{panel3}" ...>
...
</rich:panel>
<rich:panel ... rendered="#{not panel1.rendered and not panel2.rendered and not panel3.rendered}">
...
</rich:panel>
Тем не менее, в данном конкретном случае, возможно, лучше присвоить псевдониму такие длинные <c:set>
:
<c:set var="android" value="#{fn:containsIgnoreCase(header['User-Agent'], 'Android')}" scope="request" />
<c:set var="iPhone" value="#{fn:containsIgnoreCase(header['User-Agent'], 'iPhone')}" scope="request" />
<c:set var="blackBerry" value="#{fn:containsIgnoreCase(header['User-Agent'], 'BlackBerry')}" scope="request" />
<rich:panel ... rendered="#{android}">
...
</rich:panel>
<rich:panel ... rendered="#{iPhone}">
...
</rich:panel>
<rich:panel ... rendered="#{blackBerry}">
...
</rich:panel>
<rich:panel ... rendered="#{not android and not iPhone and not blackBerry}">
...
</rich:panel>
Обратите внимание, что есть более короткий способ получить заголовок запроса неявным #{header}
карта.
Ты можешь использовать
<c:choose>
<c:when test="#{fn:containsIgnoreCase(request.getHeader('User-Agent'), 'Android')}">
<rich:panel id="dlAndroid">
...
</rich:panel>
</c:when>
<c:when test="#{fn:containsIgnoreCase(request.getHeader('User-Agent'), 'iPhone')}">
<rich:panel id="dlIphone">
...
</rich:panel>
</c:when>
<c:otherwise>
<rich:panel id="dlGeneric">
...
</rich:panel>
</c:otherwise>
<c:choose>
Я думаю, что вы используете логику для определения типа браузера, который не будет меняться между пунктами на одной странице, поэтому мне кажется безопасным использовать здесь тег JSTL. JSTL-тег будет выполняться только один раз во время CREATE VIEW для страницы.
Однако если вы тестируете условие для некоторой переменной модели, то лучше обернуть каждую из панелей в ui: фрагмент как
<ui:fragment rendered="#{fn:containsIgnoreCase(request.getHeader('User-Agent'), 'Android')}">
<rich:panel id="dlAndroid">
...
</rich:panel>
<ui:fragment>
Вы можете написать эту проверку для метода управляемого компонента, чтобы он проверял, соответствует ли пользовательский агент 4-й опции. Пример:
public boolean userAgentUnknownBrowser() {
// Verify if contains the user-agent String
}
И на своей странице вы просто будете использовать метод:
<rich:panel id="dlGeneric" rendered="#{myBean.userAgentUnknownBrowser()}">