You go to the click-user or click-development mail news groups.
This is where you get support from in open source projects. Its free, friendly and usually pretty responsive.
These news groups are where other people ask questions, and where users and developers discuss ideas. Its highly recommended, so don't be shy.
To sign up to these mail groups follow the instructions at the following links:
For more details please email Malcolm Edgar.
ClickIDE is an Eclipse Web Tools Project (WTP) plugin developed by Naoki Takezoe.
To use the ClickIDE plugin, create a "Dynamic Web Project" and select the "Click" Project Faclet.
NetBeans plug-in for Click is under development by Geertjan Wielenga and Ahmed Mohombe.
Velocitywebedit is a Velocity and HTML Editor plugin for Eclipse.
After installing Velocity Web Edit configure Eclipe *.htm file associations to use Velocity Editor. Do this via the Eclipse menus: Window > Preferences > General > File Associations
There are a number of approaches you can use for generating for HTML, each with pros & cons:
This a 80/20 approach where you can quickly get stuff developed, but it may not meet all your UI style requirements.
Please note the Forms control provides many auto layout options, see the click-examples 'Form Properties' for a demonstration.
The Form control also renders the class attributes 'form', 'fields', 'errors' and 'buttons' in the form's HTML elements enabling fine grained CSS control.
As an example you could configure a form to display the labels on top, and want to increase the vertical spacing between fields. To do this simply define the CSS style in front of your form:
<style type="text/css"> td.fields { padding-top: 0.75em; } </style> $formFields are also rendered with an 'id' attribute enabling manipulation of individual fields.
This is a good approach for ensuring a common LAF across your web application, but you will have to get your hands dirty and write some code.
The Panel control support custom Velocity templates, which is an ideal way to create reusable HTML sections which you can include in many pages. Panels also support nested controls, enabling your sections to support control based behaviour.
This is easy to do and gives you good reuse across your web application.
Please see the Form velocity macros example, and also see the click-examples 'Velocity Macro' demonstration.
This gives you the ultimate control in presentation, but provides no reuse across your web application.
Please see the Form manual layout example.
Page templating is highly recommended for your web applications, as it provides numerous benefits including:
The reason you may want to abort further processing is so you can navigate directly to another page, rather than continuing to execute other control or page methods which may be time consuming. You can use this feature like a break or goto statement. For example:
public boolean onLogoutClick() { setRedirect(Logout.class); return false; }Please see the Page Navigation topic for more details.
You can even do this in your page template. Just make sure you call you call it before $form renders itself. For example:
$form.removeFields(["field11", "field15", "field22"]) $form
/com/mycorp/pages/Login.properties /com/mycorp/pages/Login_en.properties /com/mycorp/pages/Login_fr.propertiesIn your Login Page class you can use the getMessage() method to lookup request localized message strings.
public void onInit() { addModel("title", getMessage("title")); }In your HTML page template you can also access the localize message using the MessagesMap object which is added to the template using the name "messages". For example:
<h2>$messages.title</h2>You can also define Field and ActionLink labels and title values in your pages properties file using control name lookup convention. For details please see the Javadoc:
/click-control.propertiesThe field class provides a number of getMessage(String) methods which support localized strings and message formatting. Please also see the Control topic Message Properties.
input.encoding=UTF-8This will configure the Velocity runtime to use UTF-8.
You will also need to set the page headers content type to UTF-8. You can do this globally for all your applications pages using the headers element in your WEB-INF/click.xml file:
<click> .. <headers> <header name="Content-Type" value="text/html;charset=UTF-8"/> </headers> .. </click>This will set the pages HttpServletResponse "Content-Type" header to be "text/html;charset=UTF-8". Alternatively you can override the Page getContentType() method to this content type.
<click charset="UTF-8">
..
</click>
This character set is used for Velocity input encoding, and response Content-Type.
So if you specify the character set in WEB-INF/click.xml,
you don't have to create WEB-INF/velocity.properties.
If you want to use the other character set in your Velocity templates, you can override input encoding by WEB-INF/velocity.properties. See the previous topic.
Please see the Redirect After Post article for more information on this topic.
To prevent users resubmitting a form with the browser back button use the Form onSubmitCheck() method:
public class Purchase extends Page { .. public boolean onSecurityCheck() { return form.onSubmitCheck(this, "/invalid-submit.html"); } }
Please see the Click Examples 'Page Flow' for a demonstration of the Post-Redirect and submit check patterns.
However you can map URLs to different extensions using Url Rewrite Filter. It is even possible to rewrite to REST like URLs:
The Spring MVC Framework however, is not compatible with Click. The Spring framework uses a low level command pattern design like Struts and WebWork. Spring uses a DispatcherServlet to route requests to Controller objects and then passes the ModelAndView results to the rendering layer.
public interface Controller { public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception; }In Spring MVC the integration plugin points are the Controllers.
Click uses higher level design constructs focusing on Pages and Controls. Click uses its own ClickServlet for dispatching requests to Pages and Controls, which are Click's plugin points.
JSP Pages can be automatically loaded, like Velocity templates, and can be explicitly defined in the click.xml file.
<page path="customers-table.jsp" class="com.mycorp.pages.CustomersTable"/>The page's model data values are automatically added to the request as attributes so they are available in the JSP. Other Click values added as request attributes include:
Here is an example to render either a PDF or Excel document:
public MyPage extends Page {
/** The PDF report type request parameter. */
public static final String PDF = "pdf";
/** The Excel report type request parameter. */
public static final String EXCEL = "excel";
/**
* The Page#onGet renders the report to the servlet output stream.
*/
public void onGet() {
HttpServletResponse response = getContext().getResponse();
// report type is a request parameter which specifies either PDF or Excel
String reportType = getContext().getRequestParameter("reportType");
// Retrieve either a PDF or Excel document as an InputStream
InputStream inputStream = getInputStream(reportType);
// Set response headers
String mimeType = getMimeType(reportType);
response.setHeader("Content-Disposition", "attachment; filename=\"report.pdf\"");
response.setContentType(mimeType);
response.setHeader("Pragma", "no-cache");
OutputStream outputStream = null;
try {
outputStream = response.getOutputStream();
// Write out PDF Document to response stream
IOUtils.copy(inputStream, outputStream);
// By setting the Page path to null, we bypass template rendering
setPath(null);
} catch (IOException ioe) {
throw new RuntimeException(ioe);
} finally {
ClickUtils.close(outputStream);
}
}
/**
* This method should contain the logic to retrieve the report as either a PDF
* or Excel document.
*/
public InputStream getReport(String reportType) {
InputStream intputStream = null;
if (PDF.equals(reportType)) {
...
} else if (EXCEL.equals(reportType)) {
...
}
return intputStream;
}
/**
* This method returns the report mime type based on the specified report type.
*/
public String getMimeType(String reportType) {
if (PDF.equals(reportType)) {
return ClickUtils.getMimeType(".pdf");
} else if (EXCEL.equals(reportType)) {
return ClickUtils.getMimeType(".xls");
}
}
}
When the Click framework processes a request it creates a relatively small number of objects. These include a Page, a Context, a Format and a number of Contols which in turn contain some Lists, Maps and Strings.
The allocation of small/moderate numbers of short lived objects by modern JVMs is very, very fast. Please see the IBM article Urban performance legends, revisited.
The amount of actual work Click does is pretty small.
The onProcess() methods traverse a list of Controls and do simple operations.
The Control toString() methods can allocate large StringBuffers when rendering tables, but forms rarely get larger than 4,000 characters. When Click creates a new StringBuffer it attempts to determine the maximum likely size to avoid additional memory allocation and arraycopy operations.
There is not much reflection in Click at all. With reflection only really used by the Control call back listeners:
okButton.setListener(this, "onOkClicked");The next step in processing the request is rendering the response. Velocity is largely responsible for this step.
Velocity is also fast.
While hard performance numbers are difficult to come by in this area, until recently Velocity was considered to be faster than JSP. However, recent discussions on Velocity mail lists indicate that JSP compilers have improved to the point where JSP has a small performance lead over Velocity. What ever the case, JSP is very fast and so is Velocity.
If you still need to write unit tests for your Click pages, please read the documentation on the MockContainer and MockContext classes.
Click provides support for configurable LogServices and provides JdkLogService and Log4JLogService classes in Click Extras.
Please note however that Click provides support for FreeMarker with the FreemarkerTemplateService in Click Extras.
For a more comprehensive answer please see Why Click.