How to write a forrest:contract?
Intended Audience
Devs and skin developer that wants to get started with forrest:contract development. To really understand this how-to you need basic and sometimes advanced understanding of the "old fashion" skin development process.
Purpose
This setup guide will explain how to create a forrest:contract from scratch and how this forrest:contract work with the core parts of forrest.
Prerequisites
- You have a ready-to-go new seed (newSeed) based on views like described in Install views.
- This includes as well all additional plugins that are mentioned in Install views.
- Reading that how-to is as well a good idea to understand the used dir-structure in this how-to.
Steps
By working on the i18n integration for "pelt" we crossed again the whys for using views. ;-) The maintainment problem was to change the captions of the skin features (contracts) to enable support for i18n. The case is that the site2xhtml.xsl has a lot of repeating code.
For example the "last-publish"-contract could be found 2 times in the code. This is not the only contract that was (is) double in the code. The problem with that is that we needed to search the code for each caption and senseless repeat the following maintainment step of adding the <i18n:text/> -tags.
- <script language="JavaScript" - type="text/javascript">document.write("Published: " + document.lastModified);</script> + <script type="text/javascript">document.write("<i18n:text >Last + Published:</i18n:text> " + document.lastModified);
Enhance the maintainment
Now we can enhance the maintainment for the future and we give this code snippets contracts names (based on their functionality). This naming enables us to keep the contract separate from the position code itself. In xsl you would simply do:
- replace the script by <xsl:call-template name="siteinfo-last-published"/>
- and add:
<xsl:template name="siteinfo-last-published"> <script type="text/javascript"> document.write("<i18n:text >Last Published:</i18n:text> " + document.lastModified); </script> </xsl:template>
This allows us in a next maintainment just change the code of <xsl:template name="siteinfo-last-published"/> and apply it in any position where it is placed.
Explaining the blank forrest:contract
To start a new forrest:contract you can copy the 'blank.ft' from org.apache.forrest.plugin.output.viewHelper.xhtml/resources/templates. The exact file system path can be looked up at http://localhost:8888/ls.contracts.html.
The 'blank.ft' is a simple xml file with the following code which you can use to base new contracts on:
<forrest:contract xmlns:forrest="http://apache.org/forrest/templates/1.0" name="blank" type="nugget"> <!--NOTE: When using the blank template as c'n p master just search and replace 'blank' by the {contract-name}!--> <description> {contract-name} will output {contract-funtion}. This is just a blank contract, it will output *nothing*. </description> <usage><![CDATA[<forrest:contract name="blank"/>]]></usage> <forrest:template xmlns:forrest="http://apache.org/forrest/templates/1.0" format="xhtml" name="blank" inputFormat="xsl" body="false" head="false"> <xsl:stylesheet version="1.1" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <!--Add here the needed templates--> </xsl:stylesheet> </forrest:template> </forrest:contract>
The most important is the name of the contract <forrest:contract name="blank"/> . This name is the same as the file name of the contract (without file extension) blank.ft.
The <description/> tag needs to be filled in with some information that is explaining the contract to the webdesigner. The better explained the more efficient for the webdesigner to pick the right contract.
<description> siteinfo-last-published-howto will output the last published date of the site with the help of jscript. </description>
In the <usage/> tag we have to explain how the designer can use the contract in his view.
<usage><![CDATA[<forrest:contract name="siteinfo-last-published-howto"/>]]></usage>
<forrest:template name="blank" body="false" head="false"> That leads to the template attribute @body="true" and @head="false". In xhtml a contract can add content to the <body/> or/and <head/> part of <html/> . This values have to be change when adding an actual template. Besides this a xsl-template has to indicate this in the naming. A template that add content to the html body has to end with "-body"!!!
A <forrest:template /> has the son <xsl:stylesheet/> where we can create templates for the html-head and html-body. For adding content into the body of the final document change @body="true" and add:
<xsl:stylesheet version="1.1" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <!--Add here the needed templates--> <xsl:template name="blank-body"/> </xsl:stylesheet>
Create a new contract
Now lets pick up the example we started with and create a "siteinfo-last-published-howto" contract. Save the blank.ft to {project.home}/src/documentation/resources/templates/siteinfo-last-published-howto.ft.
Now the maintainment optimized code (xpath="/html/body/*") was:
<xsl:template name="siteinfo-last-published"> <script type="text/javascript"> document.write("<i18n:text >Last Published:</i18n:text> " + document.lastModified); </script> </xsl:template>
In this code we have to do the following steps to use it in our contract:
- Search and replace "siteinfo-last-published" with "siteinfo-last-publish-howto-body"
- Add a "debug string - " to the template
The contract after this steps should look like:
<xsl:template name="siteinfo-last-publish-howto-body"> debug string - <script type="text/javascript"> document.write("<i18n:text >Last Published:</i18n:text> " + document.lastModified); </script> </xsl:template>
Now we have to do some last steps in the siteinfo-last-publish-howto.ft
- Search and replace "blank" with "siteinfo-last-publish-howto"
- Add description and usage of the contract
- Set @body="true"
- Copy the maintainment optimized code to the contract.
As the result your code should look like this:
<forrest:contract xmlns:forrest="http://apache.org/forrest/templates/1.0" name="siteinfo-last-published-howto" type="nugget"> <description> siteinfo-last-published-howto will output the last published date of the site with the help of jscript. </description> <usage><![CDATA[<forrest:contract name="siteinfo-last-published-howto"/>]]></usage> <forrest:template xmlns:i18n="http://apache.org/cocoon/i18n/2.1" xmlns:forrest="http://apache.org/forrest/templates/1.0" format="xhtml" name="siteinfo-last-published-howto" inputFormat="xsl" body="true" head="false"> <xsl:stylesheet version="1.1" xmlns:i18n="http://apache.org/cocoon/i18n/2.1" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:template name="siteinfo-last-published-howto-body"> debug string - <script type="text/javascript">document.write("<i18n:text >Last Published:</i18n:text> " + document.lastModified);</script> </xsl:template> </xsl:stylesheet> </forrest:template> </forrest:contract>
Activating the contract
To see whether the new contract works we need to add it to our view. The contract usage contains the contract-tag <forrest:contract name="siteinfo-last-published-howto"/> Please see Getting started with forrest:view DSL for more details.
Further Reading
Congratulations you are now able to work with view contracts. From here we recommend to read the following How-To's:
Feedback
Please provide feedback about this document via the mailing lists.