2012-09-25 13 views
5

Tôi có một trang web bao gồm một trang phụ thay đổi theo phần tử tôi chọn trong menu dock của thủ đô (ui:include). Một số trang phụ bao gồm một thành phần tổng hợp tùy chỉnh mà tôi đã triển khai. Trang đầu tiên mà ứng dụng web hiển thị có tất cả người nghe của anh ấy hoạt động chính xác. Khi tôi thay đổi trang phụ thông qua trình đơn dock, trình lắng nghe của VideoStatusTable (trình nghe của thành phần tổng hợp) sẽ không hoạt động cho đến khi tôi làm mới trang trong trình duyệt (với F5) HOẶC nếu tôi chọn lại trang trong menu dock.Trình nghe thành phần tổng hợp chỉ hoạt động sau khi tải lại trang

Đây là trang chính đang giữ menu thanh công cụ.

<h:body style="width:100%;height:100%;position:relative;"> 
    <h:panelGroup id="contentPanelGroup"> 
     <ui:include src="#{Template.currentView.view}" /> 
    </h:panelGroup> 

    <div id="header-wrapper"> 
     <h:form id="headerForm" styleClass="titleSize" style="position:relative;height:100%;width:100%;"> 
     </h:form> 
    </div> 

    <div id="footer-wrapper"> 
       <h:form id="footerForm"> 
        <h:graphicImage name="ctec.png" library="images" style="position:absolute;left:30px;bottom:10px;"/> 
        <p:dock> 
         <p:menuitem value="#{msgs.ViewEnum_TRANSFER}" icon="#{resource['images:hard-drive-download.png']}" action="#{Template.setWindow(0)}" update=":contentPanelGroup :headerForm :msgsArea" /> 
         <p:menuitem value="#{msgs.ViewEnum_STATUS}" icon="#{resource['images:gears.png']}" action="#{Template.setWindow(1)}" update=":contentPanelGroup :headerForm :msgsArea"/> 
         <p:menuitem value="#{msgs.ViewEnum_ORGANIZATION}" icon="#{resource['images:folder.png']}" action="#{Template.setWindow(2)}" update=":contentPanelGroup :headerForm :msgsArea" /> 
         <p:menuitem value="#{msgs.ViewEnum_VALIDATION}" icon="#{resource['images:chart-bar.png']}" action="#{Template.setWindow(3)}" update=":contentPanelGroup :headerForm :msgsArea" /> 
         <p:menuitem value="#{msgs.ViewEnum_REPORT}" icon="#{resource['images:documents.png']}" action="#{Template.setWindow(4)}" update=":contentPanelGroup :headerForm :msgsArea" /> 
        </p:dock> 
       </h:form> 
    </div> 
    <p:growl id="msgsArea" life="5000"/> 
    <ui:debug/> 

</h:body> 

TemplateBean trông như thế này:

@Named(value="Template") // CDI 
@SessionScoped // CDI 
public class TemplateBean implements Serializable { 
    private static final long serialVersionUID = -8230221469543897876L; 

    private Integer    window    = 2; 

     // Some getters ... 

     // Get Window 
     public Integer getWindow() { 
      return window; 
     } 

     public void setWindow(Integer window) { 
      this.window = window; 
      FacesContext.getCurrentInstance().addMessage(
        null, 
        new FacesMessage(FacesMessage.SEVERITY_INFO, getCurrentViewTitle(), getCurrentViewTitle()) 
      ); 
      FacesContext.getCurrentInstance().addMessage(
        null, 
        new FacesMessage(FacesMessage.SEVERITY_ERROR, getCurrentViewTitle(), getCurrentViewTitle()) 
      ); 
     } 
    } 

ViewEnum (được sử dụng để lựa chọn mà xem được hiển thị):

public enum ViewEnum { 
    TRANSFER ("hard-drive-download.png", "/private/VideoTransfer.xhtml"), 
    STATUS ("gears.png", "/private/ProcessStatus.xhtml"), 
    ORGANIZATION ("folder.png", "/private/DataOrganization.xhtml"), 
    VALIDATION ("chart-bar.png", "/private/ProcessValidation.xhtml"), 
    REPORT ("documents.png", "/private/ReportGeneration.xhtml"), 
    ; 

    private String     iconFileName; 
    private String     view; 
    private StreamedContent   icon = null; 

    private ViewEnum(String iconFileName, String view) { 
     this.iconFileName = iconFileName; 
     this.view = view; 
    } 
    public String getIconFileName() { 
     return this.iconFileName; 
    } 
    public String getTranslationKey() { 
     return "ViewEnum_" + this.toString(); 
    } 
    public StreamedContent getIcon() { 
     // irrelevant code ... 
    } 
    public String getView() { 
     return this.view; 
    } 
} 

Các thành phần tùy chỉnh:

<?xml version="1.0" encoding="UTF-8"?> 
<!DOCTYPE html> 
<html xmlns="http://www.w3.org/1999/xhtml" 
     xmlns:composite="http://java.sun.com/jsf/composite" 
     xmlns:c="http://java.sun.com/jsp/jstl/core" 
     xmlns:h="http://java.sun.com/jsf/html" 
     xmlns:f="http://java.sun.com/jsf/core" 
     xmlns:ui="http://java.sun.com/jsf/facelets" 
     xmlns:p="http://primefaces.org/ui" 
     xmlns:cmpnt="http://java.sun.com/jsf/composite/component"> 

    <composite:interface componentType="videoStatusTableComponent"> 
     <composite:attribute name="value" required="true"/> 
     <composite:attribute name="selection" required="true"/> 

     <composite:attribute name="selectionListener" required="true" method-signature="void listener(org.primefaces.event.SelectEvent)"/> 
     <composite:attribute name="selectionUpdate" required="false" default="@this"/> 
     <composite:attribute name="refreshListener" required="true" method-signature="void action()"/> 
    </composite:interface> 

    <composite:implementation> 
     <p:dataTable id="cmpntVideoList" var="video" value="#{cc.attrs.value}" rowKey="#{video.key}" style="clear:both;" 
      selection="#{cc.attrs.selection}" selectionMode="single" emptyMessage="#{cc.attrs.emptyValueListMsg}"> 

      <p:ajax event="rowSelect" listener="${cc.selectionListener}" process="@this" update="${cc.attrs.selectionUpdate}"/> 

      <composite:insertFacet name="header"/> 

      <p:column headerText="Test"> 
       #{video.humanReadableVideoId} 
      </p:column> 

      <f:facet name="footer"> 
       <h:commandLink action="${cc.attrs.refreshListener}" style="float:right;"> 
        <h:graphicImage library="images" name="button-rotate-cw_16.png"/> 
        <f:ajax render="cmpntVideoList" execute="@this"/> 
       </h:commandLink> 
      </f:facet> 

     </p:dataTable> 

    </composite:implementation> 


</html> 


@FacesComponent("videoStatusTableComponent") 
public class VideoStatusTableComponent extends UINamingContainer { 

    public void selectionListener(org.primefaces.event.SelectEvent event) { 
     FacesContext context = FacesContext.getCurrentInstance(); 
     MethodExpression ajaxEventListener = (MethodExpression) getAttributes().get("selectionListener"); 
     ajaxEventListener.invoke(context.getELContext(), new Object[] { event }); 
    } 

} 

Đầu tiên sub-page (và Bean của nó), trong đó bao gồm các thành phần:

<?xml version="1.0" encoding="UTF-8"?> 
    <!DOCTYPE html> 
    <html xmlns="http://www.w3.org/1999/xhtml" 
     xmlns:h="http://java.sun.com/jsf/html" 
     xmlns:c="http://java.sun.com/jsp/jstl/core" 
     xmlns:f="http://java.sun.com/jsf/core" 
     xmlns:ui="http://java.sun.com/jsf/facelets" 
     xmlns:p="http://primefaces.org/ui" 
     xmlns:cmpnt="http://java.sun.com/jsf/composite/component"> 
    <ui:composition> 
     <h:form id="contentForm"> 
      <cmpnt:videoStatusTable id="transferingVideoList" 
       value="#{videoTransfer.tableModel}" 
       selection="#{videoTransfer.selectedTableReadyNotCompletelyTranferedVideo}" 
       selectionListener="${videoTransfer.onVideoSelection}" 
       selectionUpdate=":msgsArea" 
       refreshListener="${processStatus.refreshUncompletedVideos}" 
      > 
      </cmpnt:videoStatusTable> 
     </h:form> 
    </ui:composition> 
    </html> 



    @Named(value="videoTransfer") // CDI 
    @SessionScoped // CDI 
    public class VideoTransferBean implements Serializable { 

     private static final long serialVersionUID = -9019701853654362317L; 

     private VideoStatus       selectedTableReadyNotCompletelyTranferedVideo; 
     private VideoStatusTableModel    tableModel; 

     private List<Video>       currentlyTranferingVideos = null; 

     // Other irrelevant code... 

     public VideoStatusTableModel getTableModel() { 
      return tableModel; 
     } 

     public void setSelectedTableReadyNotCompletelyTranferedVideo(VideoStatus selectedTableReadyNotCompletelyTranferedVideo) { 
      this.selectedTableReadyNotCompletelyTranferedVideo = selectedTableReadyNotCompletelyTranferedVideo; 
     } 

     public VideoStatus getSelectedTableReadyNotCompletelyTranferedVideo() { 
      return selectedTableReadyNotCompletelyTranferedVideo; 
     } 

public void onVideoSelection(SelectEvent event) { 
     FacesMessage msg = new FacesMessage("Video Selected: " + ((VideoStatus) event.getObject()).getHumanReadableVideoId()); 
     FacesContext.getCurrentInstance().addMessage(null, msg); 
    } 
    } 

Một phụ trang đó bao gồm các thành phần tương tự (ở đây người nghe không làm việc cho đến khi tôi tải lại trang (thông qua các bến tàu hoặc nếu tôi nhấn F5 )):

<?xml version="1.0" encoding="UTF-8"?> 
<!DOCTYPE html> 
<html xmlns="http://www.w3.org/1999/xhtml" 
     xmlns:h="http://java.sun.com/jsf/html" 
     xmlns:f="http://java.sun.com/jsf/core" 
     xmlns:ui="http://java.sun.com/jsf/facelets" 
     xmlns:p="http://primefaces.org/ui" 
     xmlns:cmpnt="http://java.sun.com/jsf/composite/component" 
> 
    <ui:composition> 
     <h:form id="contentForm"> 
      <cmpnt:videoStatusTable 
       id="orphanVideoList" 
       value="#{DataOrganization.videoTableModel}" 
       selection="#{DataOrganization.selectedVideo}" 
       selectionListener="#{DataOrganization.onOrphanVideoSelection}" 
       selectionUpdate=":msgsArea" 
       refreshListener="#{DataOrganization.refreshOrphanVideos}" 
      /> 
     </h:form> 
    </ui:composition> 
</html> 
@Named(value="DataOrganization") // CDI 
@SessionScoped // CDI 
public class DataOrganizationBean implements Serializable, MonitoredBean { 

    private static final long serialVersionUID = 1686055743669628317L; 

    // Constants and variables 

    @EJB 
    private DataOrganizationEJB controller; 

    private Integer companyEntityID = null; 

    private VideoStatusTableModel videoTableModel; 
    private VideoStatus selectedVideo; 

    public void refreshOrphanVideos() { 
     setOrphanVideos(controller.getOrphanVideos(getCompanyEntityID())); 
    } 

    public void onOrphanVideoSelection(org.primefaces.event.SelectEvent event) { 
     this.setSelectedVideo(((VideoStatus) event.getObject())); 
    } 

    public VideoStatusTableModel getVideoTableModel() { 
     return videoTableModel; 
    } 

    public VideoStatus getSelectedVideo() { 
     return selectedVideo; 
    } 
    public void setSelectedVideo(VideoStatus selectedVideo) { 
     this.selectedVideo = selectedVideo; 
    } 
} 

có ai có một đầu mối về cách để tránh tải lại trang web để có được thính giả của thành phần để làm việc?

Trong XML web tôi đã đặt STATE_SAVING_METHOD cho khách hàng.

<context-param> 
    <param-name>javax.faces.STATE_SAVING_METHOD</param-name> 
    <param-value>client</param-value> 
</context-param> 

N.B .: Tôi sử dụng JSF 2.0, Glassfish 3.1.2.2, Primefaces 3.4.

Cảm ơn!

** * * CẬP NHẬT * ** *

tôi phát hiện ra vấn đề thực sự xuất phát từ các thành phần. Nếu tôi sử dụng chính xác cùng một mã mà không sử dụng các thành phần thì mọi thứ đều hoạt động tốt.

Có ai gặp phải sự cố này không?

+0

Tôi phát hiện ra vấn đề thực sự xuất phát từ các thành phần.Nếu tôi sử dụng chính xác cùng một mã mà không sử dụng các thành phần thì mọi thứ đều hoạt động tốt. Có ai gặp phải sự cố này không? – fostiguy

Trả lời

1

tôi đã cùng một vấn đề nhưng bạn thực sự nên tránh làm điều này! Bạn nên làm ít thành phần chung hơn. Nó làm việc cho tôi.