Как отобразить компонент, только если другой компонент не отображается?

У меня есть страница, к которой пользователь может получить доступ через Android, iPhone, BlackBerry или через неизвестный браузер. У меня 4 rich:panels, по одному для каждой платформы, а последняя является общей.

Код:

<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()}">
Другие вопросы по тегам