here are probably many on going debates whether to use portlets or taskflows in a WebCenter custom portal application. Usually
the main battle on which side to take in these debates are centered around which technology enables better performance.
The good news is that both
of my colleagues, Maiko Rocha and George Maggessy have posted their respective views on this topic so I will not have to further
the discussion. However, if you do plan to use portlets in a WebCenter custom portal application, this post will help you not have
the "portlet skin mismatch" issue. An example
of the presence
of the mismatch can be view from
the applications log:
The skin customsharedskin.desktop specified on
the requestMap will be used even though
the consumer's skin's styleSheetDocumentId on
the requestMap does not match
the local skin's styleSheetDocument's id. This will impact performance since
the consumer and producer stylesheets cannot be shared.
The producer styleclasses will not be compressed to avoid conflicts. A reason
the ids do not match may be
the jars are not identical on
the producer and
the consumer. For example, one might have trinidad-skins.xml's skin-additions in a jar file on
the class path that
the other does not have.
Notice that due to the mismatch
the portlet's CSS will not be able to be compressed, which will most like impact performance in the portlet's consuming portal.
The first part
of the blog will define the portlet mismatch and cover some debugging tips that can help you solve
the portlet mismatch issue. Following that I will give a complete example
of the creating, using and sharing a shared skin in both a portlet producer and
the consumer application.
Portlet Mismatch Defined
In general, when you consume/render an ADF page (or task flow) using
the ADF Portlet bridge,
the portlet (producer) would try to use
the skin
of the consumer page - this is called skin-sharing. When
the producer cannot match
the consumer skin,
the portlet would generate its own stylesheet and reference it from its markup - this is called mismatched-skin. This can happen because:
The consumer and producer use different versions
of ADF Faces, or
The consumer has additional skin-additions that
the producer doesn't have or vice-versa, or
The producer does not have
the consumer skin
For case (1) & (2) above,
the producer still uses
the consumer skin ID to render its markup. For case (3),
the producer would default to using portlet skin.
If there is a skin mis-match then there may be a performance hit because:
The browser needs to fetch this extra stylesheet (though it should be cached unless expires caching is turned off)
The generated portlet markup uses uncompressed styles resulting in a larger markup
It is often not obvious when a skin mismatch occurs, unless you look for either
of these indicators:
The log messages in
the producer log, for example:
The skin blafplus-rich.desktop specified on
the requestMap will not be used because
the styleSheetDocument id on
the requestMap does not match
the local skin's styleSheetDocument's id. It could mean
the jars are not identical. For example, one might have trinidad-skins.xml's skin-additions in a jar file on
the class path that
the other does not have.
View
the portlet markup inside
the iframe, there should be a <link> tag to
the portlet stylesheet resource like this (note
the CSS is proxied through consumer's resourceproxy):
<link rel=\"stylesheet\" charset=\"UTF-8\" type=\"text/css\" href=\"http:.../resourceproxy/portletId...252525252Fadf%252525252Fstyles%252525252Fcache%252525252Fblafplus-rich-portlet-d1062g-en-ltr-gecko.css... Using HTTP monitoring tool (eg, firebug, httpwatch), you can see a request is made to
the portlet stylesheet resource (see URL above)
There are a number
of reasons for mismatched-skin. For skin to match
the producer and consumer must match
the following configurations:
The ADF Faces version (different versions may have different style selectors)
Style Compression, this is defined in
the web.xml (default value is false, i.e. compression is ON)
Tonal styles or themes, also defined in
the web.xml via context-params
The same skin additions (jars with skin) are available for both producer and consumer. Skin additions are defined in
the trinidad-skins.xml, using
the <skin-addition> tags. These are then aggregated from all
the jar files in
the classpath. If there's any jar that exists on
the producer but not
the consumer, or vice veras, you get a mismatch.
Debugging Tips
Ensure
the style compression and tonal styles/themes match on
the consumer and producer, by looking at
the web.xml documents for
the consumer & producer applications
It is bit more involved to determine if
the jars match. However, you can enable
the Trinidad logging to show which skin-addition it is processing. To enable this feature, update
the logging.xml log level
of both
the producer and consumer WLS to FINEST. For example, in
the case
of the WebLogic server used by JDeveloper:
$JDEV_USER_DIR/system<version number>/DefaultDomain/config/fmwconfig/servers/DefaultServer/logging.xml
Add a new entry:
<logger name="org.apache.myfaces.trinidadinternal.skin.SkinUtils" level="FINEST"/>
Restart WebLogic. Run
the consumer page, you should see
the following logging in both
the consumer and producer log files. Any entries that don't match is
the cause
of the mismatch.
The following is an example
of what
the log will produce with this setting:
[SRC_CLASS: org.apache.myfaces.trinidadinternal.skin.SkinUtils] [APP: WebCenter] [SRC_METHOD: _getMetaInfSkinsNodeList]
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/announcement-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/calendar-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/custComps-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/forum-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/page-service-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/peopleconnections-kudos-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/peopleconnections-wall-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/portlet-client-adf-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/rtc-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/serviceframework-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/smarttag-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.skin/in1ar8/APP-INF/lib/spaces-service-skins.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/oracle.webcenter.composer/3yo7j/WEB-INF/lib/custComps-skin.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/adf.oracle.domain.webapp/q433f9/WEB-INF/lib/adf-richclient-impl-11.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/adf.oracle.domain.webapp/q433f9/WEB-INF/lib/dvt-faces.jar!/META-INF/trinidad-skins.xml
Processing skin URL:zip:/tmp/_WL_user/adf.oracle.domain.webapp/q433f9/WEB-INF/lib/dvt-trinidad.jar!/META-INF/trinidad-skins.xml
The Complete Example
The first step is to create
the shared library.
The WebCenter documentation covering this is located here in section 15.7. In addition, our ADF guru Frank Nimphius also covers this in hes blog. Here are my steps (in JDeveloper) to create
the skin that will be used as
the shared library for both
the portlet producer and consumer.
Create a new Generic Application
Give application name (i.e. MySharedSkin)
Give a project name (i.e. MySkinProject)
Leave Project Technologies blank (none selected), and click Finish
Create
the trinidad-skins.xml
Right-click on
the MySkinProject node in
the Application Navigator and select "New"
In
the New Galley, click on "General", select "File" from
the Items, and click OK
In
the Create File dialog, name the file trinidad-skins.xml, and (IMPORTANT) give
the directory path to MySkinProject\src\META-INF
In
the trinidad-skins.xml, complete
the skin entry. for example:
<?xml version="1.0" encoding="windows-1252" ?> <skins xmlns="http://myfaces.apache.org/trinidad/skin"> <skin> <id>mysharedskin.desktop</id> <family>mysharedskin</family> <extends>fusionFx-v1.desktop</extends> <style-sheet-name>css/mysharedskin.css</style-sheet-name> </skin> </skins>
Create CSS file
In
the Application Navigator, right click on
the META-INF folder (where
the trinidad-skins.xml is located), and select "New"
In
the New Gallery, select Web-Tier-> HTML, CSS File from
the the Items and click OK
In
the Create Cascading Style Sheet dialog, give
the name (i.e. mysharedskin.css)
Ensure that
the Directory path is
the under
the META-INF (i.e. MySkinProject\src\META-INF\css)
Once
the new CSS opens in
the editor, add in a style selector. For example, this selector will style
the background
of a particular panelGroupLayout:
af|panelGroupLayout.customPGL{ background-color:Fuchsia; }
Create
the MANIFEST.MF (used for deployment JAR)
In
the Application Navigator, right click on
the META-INF folder (where
the trinidad-skins.xml is located), and select "New"
In
the New Galley, click on "General", select "File" from
the Items, and click OK
In
the Create File dialog, name the file MANIFEST.MF, and (IMPORTANT) ensure that
the directory path is to MySkinProject\src\META-INF
Complete
the MANIFEST.MF, where
the extension name is
the shared library name
Manifest-Version: 1.1 Created-By: Martin Deh Implementation-Title: mysharedskin Extension-Name: mysharedskin.lib.def Specification-Version: 1.0.1 Implementation-Version: 1.0.1 Implementation-Vendor: MartinDeh
Create new Deployment Profile
Right click on
the MySkinProject node, and select New
From
the New Gallery, select General->Deployment Profiles, Shared Library JAR File from Items, and click OK
In
the Create Deployment Profile dialog, give name (i.e.mysharedskinlib) and click OK
In
the Edit JAR Deployment dialog, un-check Include Manifest File option
Select Project Output->Contributors, and check Project Source Path
Select Project Output->Filters, ensure that all items under
the META-INF folder are selected
Click OK to exit
the Project Properties dialog
Deploy
the shared lib to WebLogic (start server before steps)
Right click on MySkin Project and select Deploy
For this example, I will deploy to JDeverloper WLS
In
the Deploy dialog, select Deploy to Weblogic Application Server and click Next
Choose IntegratedWebLogicServer and click Next
Select Deploy to selected instances in
the domain radio, select Default Server (note: server must be already started), and ensure Deploy as a shared Library radio is selected
Click Finish
Open
the WebLogic console to see
the deployed shared library
The following are
the steps to create a simple test Portlet
Create a new WebCenter Portal - Portlet Producer Application
In
the Create Portlet Producer dialog, select default settings and click Finish
Right click on
the Portlets node and select New
IIn
the New Gallery, select Web-Tier->Portlets, Standards-based Java Portlet (JSR 286) and click OK
In
the General Portlet information dialog, give portlet name (i.e. MyPortlet) and click Next 2 times, stopping at Step 3
In
the Content Types, select
the "view" node, in
the Implementation Method, select
the Generate ADF-Faces JSPX radio and click Finish
Once
the portlet code is generated, open
the view.jspx in
the source editor
Based on
the simple CSS entry, which sets
the background color
of a panelGroupLayout, replace the <af:form/> tag with the example code
<af:form> <af:panelGroupLayout id="pgl1" styleClass="customPGL"> <af:outputText value="background from shared lib skin" id="ot1"/> </af:panelGroupLayout> </af:form>
Since this portlet is to use
the shared library skin, in
the generated trinidad-config.xml, remove both
the skin-family tag and
the skin-version tag
In
the Application Resources view, under Descriptors->META-INF, double-click to open
the weblogic-application.xml
Add a library reference to
the shared skin library (note:
the library-name must match
the extension-name declared in
the MANIFEST.MF):
<library-ref> <library-name>mysharedskin.lib.def</library-name> </library-ref>
Notice that a reference to oracle.webcenter.skin exists. This is important if this portlet is going to be consumed by a WebCenter Portal application. If this tag is not present,
the portlet skin mismatch will happen.
Configure
the portlet for deployment
Create Portlet deployment WAR
Right click on
the Portlets node and select New
In
the New Gallery, select Deployment Profiles, WAR file from Items and click OK
In
the Create Deployment Profile dialog, give name (i.e. myportletwar), click OK
Keep all
of the defaults, however, remember
the Context Root entry (i.e. MyPortlet4SharedLib-Portlets-context-root, this will be needed to obtain
the producer WSDL URL)
Click OK, then OK again to exit from
the Properties dialog
Since
the weblogic-application.xml has to be included in
the deployment,
the portlet must be deployed as a WAR, within an EAR
In
the Application dropdown, select Deploy->New Deployment Profile...
By default EAR File has been selected, click OK
Give Deployment Profile (EAR) a name (i.e. MyPortletProducer) and click OK
In
the Properties dialog, select Application Assembly and ensure that
the myportletwar is checked
Keep all
of the other defaults and click OK
For this demo, un-check
the Auto Generate ..., and all
of the Security Deployment Options, click OK
Save All
In
the Application dropdown, select Deploy->MyPortletProducer
In
the Deployment Action, select Deploy to Application Server, click Next
Choose IntegratedWebLogicServer and click Next
Select Deploy to selected instances in
the domain radio, select Default Server (note: server must be already started), and ensure Deploy as a standalone Application radio is selected
The select deployment type (identifying
the deployment as a JSR 286 portlet) dialog appears. Keep default radio "Yes" selection and click OK
Open
the WebLogic console to see
the deployed Portlet
The last step is to create
the test portlet consuming application. This will be done using
the OOTB WebCenter Portal - Framework Application.
Create
the Portlet Producer Connection
In
the JDeveloper Deployment log, copy
the URL
of the portlet deployment (i.e. http://localhost:7101/MyPortlet4SharedLib-Portlets-context-root
Open a browser and paste in
the URL.
The Portlet information page should appear. Click on
the WSRP v2 WSDL link
Copy
the URL from
the browser (i.e. http://localhost:7101/MyPortlet4SharedLib-Portlets-context-root/portlets/wsrp2?WSDL)
In
the Application Resources view, right click on
the Connections folder and select New Connection->WSRP Connection
Give
the producer a name or accept
the default, click Next
Enter (paste in)
the WSDL URL, click Next
If connection to Portlet is succesful, Step 3 (Specify Additional ...) should appear. Accept defaults and click Finish
Add
the portlet to a test page
Open
the home.jspx. Note in
the visual editor,
the orange dashed border, which identifies
the panelCustomizable tag.
From
the Application Resources. select
the MyPortlet portlet node, and drag and drop
the node into
the panelCustomizable section. A Confirm Portlet Type dialog appears, keep default ADF Rich Portlet and click OK
Configure the portlet to use
the shared skin library
Open
the weblogic-application.xml and add
the library-ref entry (mysharedskin.lib.def) for
the shared skin library. See create portlet example above for
the steps
Since by default,
the custom portal using a managed bean to (dynamically) determine
the skin family,
the default trinidad-config.xml will need to be altered
Open
the trinidad-config.xml in
the editor and replace
the EL (preferenceBean) for
the skin-family tag, with mysharedskin (this is
the skin-family named defined in
the trinidad-skins.xml)
Remove
the skin-version tag
Right click on
the index.html to test
the application
Notice that the JDeveloper log view does not have any reporting
of a skin mismatch. In addition, since I have configured
the extra logging outlined in debugging section above, I can see
the processed skin jar in both
the producer and consumer logs:
<SkinUtils> <_getMetaInfSkinsNodeList> Processing skin URL:zip:/JDeveloper/system11.1.1.6.38.61.92/DefaultDomain/servers/DefaultServer/upload/mysharedskin.lib.def/
[email protected]/app/mysharedskinlib.jar!/META-INF/trinidad-skins.xml