4.7 Implementation of the Remaining Use Cases

Now we can return to our implementation and focus our attention on the rest of the use cases from Chapter 3. To better understand their implementation, however, we should take another look at the input screen and the processing logic of the corresponding use case before taking any practical steps.

4.7.1 Editing Campaigns and Creating New Campaigns

The use cases Edit campaigns and Create new campaign have the same input screen and can therefore be managed with the same controller class. The only difference between the two use cases is that the first case involves the editing of the existing Campaign object, which means that the current values of the properties must be displayed in the input screen. If a new campaign is being set up, a new Campaign object must be created and its properties set with new values via the input screen. Furthermore, we must manage the processing mode that determines whether a new campaign should be created or an existing campaign edited. The management of the Campaign object and the processing mode is implemented in the class CampaignProducer (see ‎Listing  4-19).

The CampaignProducer Class

package de.dpunkt.myaktion.data;

import de.dpunkt.myaktion.model.Campaign;
import javax.enterprise.context.SessionScoped;
import javax.inject.Named;
import java.io.Serializable;

@SessionScoped
@Named
public class CampaignProducer implements Serializable {
	private static final long serialVersionUID = -1828660647917534556L;

	private enum Mode {
		EDIT, ADD
	}
	private Campaign campaign;
	private Mode mode;

	public Campaign getSelectedCampaign() {
		return campaign;
	}
	public void setSelectedCampaign(Campaign campaign) {
		this.campaign = campaign;
	}
	public boolean isAddMode() {
		return mode == Mode.ADD;
	}
	public void prepareAddCampaign() {
		this.campaign = new Campaign();
		this.mode = Mode.ADD;
	}
	public void prepareEditCampaign(Campaign campaign) {
		this.campaign = campaign;
		this.mode = Mode.EDIT;
	}
}

Listing 4-19     CampaignProducer class

The processing mode is determined via an inner enumeration type (enum Mode). It can take the values EDIT or ADD. For this reason, alongside the attribute of type Campaign, the class also possesses an attribute of the type  CampaignProducer.Mode.

An object of the class can be injected in a controller bean, making its data is available for the whole session. The controller can carry out the initialization of the bean depending on the current processing mode either via the method prepareAddCampaign or the method prepareEditCampaign. In the first case, a new Campaign object is created; in the second case, the object to be edited is passed. The rest of the getter and setter methods enable the attributes of the bean to be accessed.

The use case controller class EditCampaignController (see Listing 4-20) uses both the class CampaignProducer and the class CampaignListProducer, which manages our example data. This is necessary because new or edited campaigns must be added to the list of campaigns. It’s at this point that dependency injection comes into play. We declare an attribute of the type CampaignListProducer and an attribute of the type CampaignProducer in the controller class and and add the annotation @Inject to each. At runtime, the application server will now ensure that we have access to the bean via the declared variables. Since the context is set to @SessionScoped in both classes, we can be sure that all controller beans of the same user session are accessing the same bean.

The EditCampaignController Class

package de.dpunkt.myaktion.controller;

import de.dpunkt.myaktion.data.CampaignListProducer;
import de.dpunkt.myaktion.data.CampaignProducer;
import javax.enterprise.context.SessionScoped;
import javax.inject.Inject;
import javax.inject.Named;
import java.io.Serializable;

@SessionScoped
@Named
public class EditCampaignController implements Serializable {
	private static final long serialVersionUID = 2815796004558360299L;

	@Inject
	private CampaignListProducer campaignListProducer;
	@Inject
	private CampaignProducer campaignProducer;

	public String doSave() {
		if (campaignProducer.isAddMode()) {
			campaignListProducer.getCampaigns().add(
					campaignProducer.getSelectedCampaign());
		}
		return Pages.LIST_CAMPAIGNS;
	}
	public String doCancel() {
		return Pages.LIST_CAMPAIGNS;
	}
}

Listing 4-20     EditCampaignController class

Our controller classes contain two methods for controlling the flow. On the one hand, after editing or creating a campaign, we must be able to save it. This is accomplished by the doSave method. The logic of the method must take the current mode into account. If we are editing an existing campaign, there is nothing more to be done. However, if a new campaign is created, this must be admitted to the list of campaigns via the available ListCampaignProducer bean. In both cases, the user is redirected to the view LIST_CAMPAIGNS afterwards.

The option to cancel the process is also provided for. This is accomplished by the method doCancel. Here, all we need to is to navigate back to the view campaignList.

The editCampaign.xhtml Facelet

If we look at the layouts of the input screens in Figures 3-14 and 3-15 in Chapter 3, we see that a tab panel  was provided. In the layout, we have arranged the general campaign data and the bank account details in different tabs. However, we have not yet talked about how such a component would work – and we would not find it in the JSF standard, even if we wanted to! This is a typical motive for using an additional JSF library. We will demonstrate the use of PrimeFaces for this purpose in section ‎4.9. For the moment, we’ll do without tabs and represent the two sets of data one under the other in two grids (tag <h:panelGrid>). As for every other Facelet of the application, we also require a number of texts, which will be displayed on the page according to the language preferences set in the browser. We will modify the properties file messages_en.properties with the following input:

# editCampaign.xhtml
editCampaign.add_new_campaign=Add a new campaign
editCampaign.edit_campaign=Edit campaign
editCampaign.general=General
editCampaign.name=Name
editCampaign.campaign_name_validation=The name of a campaign must be at least 4 and max. 30 characters long.
editCampaign.name_of_the_campaign=Name of the campaign
editCampaign.target_amount=Target amount
editCampaign.target_amount_validation=The target amount for the campaign must be at least 10 Euros.
editCampaign.donation_amount=Donation amount
editCampaign.donation_amount_validation=The donation amount must be at least 1 Euro.
editCampaign.bank_account=Bank account
editCampaign.iban=IBAN
editCampaign.name_of_bank=Name of Bank
editCampaign.save=Save

In addition, we will add the following lines to the file messages_de.properties:

# editCampaign.xhtml
editCampaign.add_new_campaign=Neue Aktion anlegen
editCampaign.edit_campaign=Aktionsdaten editieren
editCampaign.general=Allgemein
editCampaign.name=Name
editCampaign.campaign_name_validation=Der Name einer Aktion muss min. 4 und darf max. 30 Zeichen lang sein.
editCampaign.name_of_the_campaign=Name der Spendenaktion
editCampaign.target_amount=Spendenziel
editCampaign.target_amount_validation=Das Spendenziel für die Aktion muss min. 10 Euro sein.
editCampaign.donation_amount=Spendenbetrag
editCampaign.donation_amount_validation=Der Spendenbetrag muss min. 1 Euro sein.
editCampaign.bank_account=Bankverbindung
editCampaign.iban=IBAN
editCampaign.name_of_bank=Name der Bank
editCampaign.save=Speichern
editCampaign.cancel=Abbrechen

‎Listing  4-21 shows the implementation of the Facelet.

<html xmlns="http://www.w3.org/1999/xhtml"
	xmlns:ui="http://xmlns.jcp.org/jsf/facelets"
	xmlns:f="http://xmlns.jcp.org/jsf/core"
	xmlns:h="http://xmlns.jcp.org/jsf/html">
<body>
	<f:view contracts="#{view.locale.language}">
		<ui:composition template="/template.xhtml">
			<ui:define name="content">
				<h1>#{campaignProducer.addMode ?
					msg['editCampaign.add_new_campaign'] :
					msg['editCampaign.edit_campaign']}</h1>
				<h:form>
					<h:panelGrid columns="3">
						<f:facet name="header">
							#{msg['editCampaign.general']}
						</f:facet>
						<h:outputLabel value="#{msg['editCampaign.name']}:">
						</h:outputLabel>
						<h:inputText id="a_name"
							value="#{campaignProducer.selectedCampaign.name}"
							validatorMessage=
								"#{msg['editCampaign.campaign_name_validation']}">
							<f:passThroughAttribute name="placeholder"
								value=
									"#{msg['editCampaign.name_of_the_campaign']}" 
							/>
							<f:validateRequired />
							<f:validateLength minimum="4" maximum="30" />
						</h:inputText>
						<h:message for="a_name" style="color: red;" />
						<h:outputLabel value=
							"#{msg['editCampaign.target_amount']}:">
						</h:outputLabel>
						<h:panelGroup>
							<h:inputText id="a_targetAmount"
								value=
									"#{campaignProducer
										.selectedCampaign.targetAmount}"
								validatorMessage=
									"#{msg['editCampaign
										.target_amount_validation']}">
								<f:convertNumber maxFractionDigits="2" 
									minFractionDigits="2" />
								<f:validateRequired />
								<f:validateDoubleRange minimum="10.0" />
							</h:inputText> EUR
						</h:panelGroup>
						<h:message for="a_targetAmount" style="color: red;" />
						<h:outputLabel value=
							"#{msg['editCampaign.donation_amount']}:">
						</h:outputLabel>
						<h:panelGroup>
							<h:inputText id="a_donationMinimum"
								value=	"#{campaignProducer
									.selectedCampaign.donationMinimum}"
								validatorMessage=
									"#{msg['editCampaign.
										donation_amount_validation']}">
								<f:convertNumber maxFractionDigits="2" 
									minFractionDigits="2" />
								<f:validateRequired />
								<f:validateDoubleRange minimum="1.0" />
							</h:inputText> EUR
						</h:panelGroup>
						<h:message for="a_donationMinimum" 
							style="color: red;" />
					</h:panelGrid>
					<h:panelGrid columns="3">
						<f:facet name="header">
							#{msg['editCampaign.bank_account']}
						</f:facet>
						<h:outputLabel value=
							"#{msg['editCampaign.name']}:">
						</h:outputLabel>
						<h:inputText id="b_name"
							value=
								"#{campaignProducer
									.selectedCampaign.account.name}">
							<f:validateRequired />
						</h:inputText>
						<h:message for="b_name" style="color: red;" />
						<h:outputLabel value=
							"#{msg['editCampaign.iban']}:">
						</h:outputLabel>
						<h:inputText id="b_iban"
							value=	"#{campaignProducer
										.selectedCampaign.account.iban}">
							<f:validateRequired />
						</h:inputText>
						<h:message for="b_iban" style="color: red;" />
						<h:outputLabel value=
							"#{msg['editCampaign.name_of_bank']}:">
						</h:outputLabel>
						<h:inputText id="b_name_bank"
							value=	"#{campaignProducer
										.selectedCampaign.account.nameOfBank}">
							<f:validateRequired />
						</h:inputText>
						<h:message for="b_name_bank" style="color: red;" />
					</h:panelGrid>
					<h:commandButton value="#{msg['editCampaign.save']}"
						action="#{editCampaignController.doSave}" />
					<h:commandButton value="#{msg['editCampaign.cancel']}"
						immediate="true" action=
							"#{editCampaignController.doCancel}" />
				</h:form>
			</ui:define>
		</ui:composition>
	</f:view>
</body>
</html> 

Listing 4-21     editCampaign.xhtml Facelet

As with previous Facelets, we will save the file editCampaign.xhtml in our project structure in the directory src\main\webapp.

In the Facelet, we can easily recognize the binding of the attributes of the Campaign object to the input fields using EL expressions. The respective Campaign object is referenced in the EL expression via the selectedCampaign attribute of the CampaignProducer bean. The methods doSave and doCancel belonging to the controller class editCampaignController are registered with both buttons of the Facelet as action methods using EL expressions.

Adjusting the Existing Facelets and Classes

We have now created the Facelet and the controller for both use cases. All we are missing is integration with the use case Display and edit campaign. To do this, we must edit the class ListCampaignsController, which will reference the editCampaign.xhtml using the methods doEditCampaign and doAddCampaign. In both methods, the relevant campaign (a new campaign, or one selected by the user) and the processing mode must be set. We therefore require a reference to the CampaignProducer bean. To create this, we will declare an instance variable of the type CampaignProducer in the ListCampaignsController class and provide it with the annotation @Inject. This causes the application server to provide a bean for the desired reference at runtime.

@Inject
private CampaignProducer campaignProducer;

We must not forget to add the import command for the annotation @Inject.

import javax.inject.Inject;

We will now use this reference in the methods doEditCampaign and doAddCampaign to implement the logic. We can suppress the logging to standard output using System.out.println.

...
public String doAddCampaign() {
	campaignProducer.prepareAddCampaign();
	return Pages.EDIT_CAMPAIGN;
}
public String doEditCampaign(Campaign campaign) {
	campaignProducer.prepareEditCampaign(campaign);
	return Pages.EDIT_CAMPAIGN; 
}
...

When editing an existing campaign, the Campaign object to be edited is passed as an argument. When creating a campaign, a new object is created in the prepareAddCampaign method of the CampaignProducer class.

The use cases Edit campaign and Create new campaign can now be run. Try them out for yourself! Please don’t forget that WildFly must be started and that we must redeploy the most recent version of the project with Maven. For comparison purposes, Fig. 4-3 shows the view that should appear for adding a new campaign. You can see the inline label in the Name input field.

 

Fig. 4-3     View for creating a new campaign

At this point, it’s essential for us to see how the page behaves when an input is entered incorrectly. In order to receive an error message, we will simply leave the input fields empty and click on the Save button. The result is shown in Fig. 4-4. Firstly, we see the standard error message (j_idt8:*). Secondly, we see the error messages that we set explicitly via the validatorMessage attribute when creating the text inputs (<h:inputText>). Naturally, the error messages are shown in the language set in the browser.

Fig. 4-4     The form for creating a new campaign, with error messages

4.7.2 Displaying the List of Donations

With the next use case, Display list of donors, we want to see who has already donated to a campaign. If we look at our domain classes, we see the relationship between campaign and donation is particularly important for this use case. A campaign can have no donations or multiple donations and donations are managed as a list for each campaign. Let’s look first at the controller class and then the Facelet.

The ListDonationsController Controller Class

As laid out in Chapter 3, we only need a confirmation button on the page that will redirect the user back to the campaign list homepage when the button is pressed. This logic will be implemented in the doOk method.

package de.dpunkt.myaktion.controller;

import de.dpunkt.myaktion.model.Donation.Status;
import javax.enterprise.context.SessionScoped;
import javax.inject.Named;
import java.io.Serializable;

@SessionScoped
@Named
public class ListDonationsController implements Serializable {
	private static final long serialVersionUID = 437878972432L;

	public String doOk() {
		return Pages.LIST_CAMPAIGNS;
	}
}

Listing 4-22     ListDonationsController class

The listDonations.xhtml Facelet

For the implementation of this Facelet, we will refer to the Facelet listCampaigns.xhtml in ‎Listing  4-21, since it also presents data from a list in tabular form. Here, again, we will use the h:dataTable tag within the form (see ‎Listing  4-23). Before that, we will modify our two properties files as we’ve become accustomed to doing. The file messages_en.properties will be supplemented with the following rows:

# listDonations.xhtml
listDonations.donation_list=List of donations
listDonations.beneficiary_account=Beneficiary account
listDonations.iban=IBAN
listDonations.name=Name
listDonations.name_of_bank=Name of Bank
listDonations.donation_amount=Donation amount
listDonations.receipt=Receipt
listDonations.yes=yes
listDonations.no=no
listDonations.status=Status
listDonations.status.TRANSFERRED=transferred
listDonations.status.IN_PROCESS=in process

We will then add the corresponding German terms to the messages_de.properties file:

# listDonations.xhtml
listDonations.donation_list=Spenderliste
listDonations.beneficiary_account=Empfängerkonto
listDonations.iban=IBAN
listDonations.name=Name
listDonations.name_of_bank=Name der Bank
listDonations.donation_amount=Spendenbetrag
listDonations.receipt=Quittung
listDonations.yes=ja
listDonations.no=nein
listDonations.status=Status
listDonations.status.TRANSFERRED=überwiesen
listDonations.status.IN_PROCESS=in Bearbeitung

We can now proceed with the implementation of the Facelet listDonations.xhtml:

<html xmlns="http://www.w3.org/1999/xhtml"
	xmlns:ui="http://xmlns.jcp.org/jsf/facelets"
	xmlns:f="http://xmlns.jcp.org/jsf/core"
	xmlns:h="http://xmlns.jcp.org/jsf/html">
<body>
	<f:view contracts="#{view.locale.language}">
		<ui:composition template="/template.xhtml">
			<ui:define name="content">
				<h1>#{msg['listDonations.donation_list']}
					'#{campaignProducer.selectedCampaign.name}'
				</h1>
				<h2>#{msg['listDonations.beneficiary_account']}:
					#{campaignProducer.selectedCampaign.account.name},
					#{msg['listDonations.iban']}:
					#{campaignProducer.selectedCampaign.account.iban},
					#{campaignProducer
						.selectedCampaign.account.nameOfBank}
				</h2>
				<h:form>
					<h:dataTable value=
							"#{campaignProducer.selectedCampaign.donations}"
						var="donation">
						<h:column>
							<f:facet name="header">
								#{msg['listDonations.name']}
							</f:facet>
							<h:outputText value="#{donation.donorName}" />
						</h:column>
						<h:column>
							<f:facet name="header">
								#{msg['listDonations.iban']}
							</f:facet>
							<h:outputText value="#{donation.account.iban}" />
						</h:column>
						<h:column>
							<f:facet name="header">
								#{msg['listDonations.name_of_bank']}
							</f:facet>
							<h:outputText value=
								"#{donation.account.nameOfBank}" />
						</h:column>
						<h:column>
							<f:facet name="header">
								#{msg['listDonations.donation_amount']}
							</f:facet>
							<h:outputText value="#{donation.amount}">
								<f:convertNumber type="currency" 
									currencyCode="EUR" />
							</h:outputText>
						</h:column>
						<h:column>
							<f:facet name="header">
								#{msg['listDonations.receipt']}
							</f:facet>
							<h:outputText value=
								"#{donation.receiptRequested ? 
									msg['listDonations.yes'] 
										: msg['listDonations.no']}" />
						</h:column>
						<h:column>
							<f:facet name="header">
								#{msg['listDonations.status']}
							</f:facet>
							<h:outputText value=
									"#{msg['listDonations.status.TRANSFERRED']}"
								rendered="#{donation.status == 'TRANSFERRED'}" />
							<h:outputText value=
									"#{msg['listDonations.status.IN_PROCESS']}"
								rendered="#{donation.status == 'IN_PROCESS'}" />
						</h:column>
					</h:dataTable>
					<h:commandButton value="OK"
						action="#{listDonationsController.doOk}" />
				</h:form>
			</ui:define>
		</ui:composition>
	</f:view>
</body>
</html> 

Listing 4-24     listDonations.xhtml Facelet

Most of the tags used have already been explained in the context of understanding the list of campaigns, so we can refrain from giving a fresh explanation here. The following two constructs however, are new:

<h:outputText value="#{donation.receiptRequested ? 
			msg['listDonations.yes']: msg['listDonations.no']}" />

Here, the conditional operator used in the EL expression should be well-known to the reader from the Java programming language. The value taken by the value attribute of the <h:outputText> tag is determined depending on a boolean expression. The boolean expression is placed before the question mark and, in our case, is implemented by the receiptRequested attribute of type boolean belonging to the class Donation. If the value is true, the value after the question mark and up to the colon is used; if not, the value after the colon is used. With regards to content, this is an elegant way to display whether or not a donation receipt is required.

The second construct also makes use of the expressive power of the expression language:

<h:outputText value=#{msg['listDonations.status.TRANSFERRED']}"
			rendered="#{donation.status == 'TRANSFERRED'}" />
<h:outputText value="#{msg['listDonations.status.IN_PROCESS']}"
			rendered="#{donation.status == 'IN_PROCESS'}" />

The rendered attribute of the <h:outputText> tag expects a boolean value and uses it to control the display. The text will be displayed only if the value is set to true. For the attribute’s value, we use an EL expression that contains a comparison operation to check which status is set in the donation to be displayed. Only one of the comparisons can ever be successful, with the result that only one of the texts can actually be displayed.

Adjusting the Existing Classes

The ListDonationsController bean requires the selected Campaign object, which should have been stored in the CampaignProducer beforehand.

This must occur in the doListDonations method of the ListCampaignsController class, since this is where the selected Campaign object is passed as an argument.

In this method, therefore, we once again require a reference to the CampaignProducer bean to enable us to pass it the chosen campaign. The doListDonations method of the ListCampaignsController class can be implemented as follows:

public String doListDonations(Campaign campaign) {
	campaignProducer.setSelectedCampaign(campaign);
	return Pages.LIST_DONATIONS;
}

Nothing more needs to be adjusted. We can try out the newly implemented use case immediately following the obligatory Maven build. As an example, ‎Fig.4-5 shows the list of donors for the first campaign. The donations are represented in tabular form as desired (tag <h:dataTable>).

 

Fig. 4-5     View of the donor list for the campaign “Shirts for the U19 Team”

4.7.3 Editing a Donation Form

We’ll now look at the use case Edit donation form. This use case should enable the user to manage the address (URL) of the donation form and a number of settings pertaining to how it is displayed. The background color and the color of the text can be adjusted for the form and are included in the address as HTTP GET parameters. We will begin by re-implementing the controller. Once this is done, we’ll move to the Facelet.

The EditDonationFormController Class

The controller bean manages the corresponding campaign and the previously-named properties: address (url), background color (bgColor) and text color (textColor). The URL is dynamically created from the other properties. Leaving out the color properties and the campaign ID (campaignId), we will first hard-code the URL. The values of the properties will be encoded in the HTTP GET request – which is specified by the URL – as values of the request parameters bgColor, textColor and campaignId.

The class in ‎Listing  4-24 has the usual getter and setter methods and requires access to the chosen campaign in order to generate the URL. This access is obtained once again via the CampaignProducer bean, which we will inject in the controller for this purpose. In line with the description of the use case in Chapter 3, only one confirmation button is required in the view. The logic will be implemented in the controller using the method doOk(), at which point nothing else happens except the redirection of the user back to the list of campaigns.

package de.dpunkt.myaktion.controller;

import de.dpunkt.myaktion.data.CampaignProducer;
import javax.enterprise.context.SessionScoped;
import javax.faces.context.FacesContext;
import javax.inject.Inject;
import javax.inject.Named;
import javax.servlet.http.HttpServletRequest;
import java.io.Serializable;

@SessionScoped
@Named
public class EditDonationFormController implements Serializable {
	private static final long serialVersionUID = -4210085664588144340L;
	private String textColor = "000000";
	private String bgColor = "ffffff";
	@Inject
	private CampaignProducer campaignProducer;

	public String doOk() {
		return Pages.LIST_CAMPAIGNS;
	}
	public String getUrl() {
		return "http://localhost:8080/my-aktion/donateMoney.jsf?bgColor="
				+ bgColor + "&textColor=" + textColor + "&campaignId="
				+ campaignProducer.getSelectedCampaign().getId();
	}
	public String getTextColor() {
		return textColor;
	}
	public void setTextColor(String textColor) {
		this.textColor = textColor;
	}
	public String getBgColor() {
		return bgColor;
	}
	public void setBgColor(String bgColor) {
		this.bgColor = bgColor;
	}
}

Listing 4-24     EditDonationFormController class

The editDonationForm.xhtml Facelet

Next, we will determine the language-dependent text for the Facelet. To do this, we insert the following lines in the properties file messages_en.properties:

# editDonationForm.xhtml
editDonationForm.edit_donation_form=Edit donation form
editDonationForm.background_color=Background color
editDonationForm.text_color=Text color
editDonationForm.form_url=Form URL

The properties file messages_de.properties should then be supplemented with the following entries:

# editDonationForm.xhtml
editDonationForm.edit_donation_form=Spendenformular bearbeiten
editDonationForm.background_color=Hintergrundfarbe
editDonationForm.text_color=Textfarbe
editDonationForm.form_url=URL des Formulars

‎Listing  4-25 shows the implementation of the Facelet for editing the properties of the donation form.

<html xmlns="http://www.w3.org/1999/xhtml"
	xmlns:ui="http://xmlns.jcp.org/jsf/facelets"
	xmlns:f="http://xmlns.jcp.org/jsf/core"
	xmlns:h="http://xmlns.jcp.org/jsf/html">
<body>
	<f:view contracts="#{view.locale.language}">
		<ui:composition template="/template.xhtml">
			<ui:define name="content">
				<h1>#{msg['editDonationForm.edit_donation_form']}</h1>
				<h:form>
					<h:panelGrid columns="2">
						<h:outputLabel
							value=
								"#{msg['editDonationForm.background_color']}:">
						</h:outputLabel>
						<h:selectOneMenu id="bgColor"
								value="#{editDonationFormController.bgColor}">
							<f:selectItem itemLabel="#FFFFFF" 
								itemValue="FFFFFF" />
							<f:selectItem itemLabel="#000000" 
								itemValue="000000" />
							<f:selectItem itemLabel="#FF0000" 
								itemValue="FF0000" />
							<f:ajax render="url" />
						</h:selectOneMenu>
						<h:outputLabel value=
							"#{msg['editDonationForm.text_color']}:">
						</h:outputLabel>
						<h:selectOneMenu id="textColor"
							value="#{editDonationFormController.textColor}">
							<f:selectItem itemLabel="#FFFFFF" 
								itemValue="FFFFFF" />
							<f:selectItem itemLabel="#000000" 
								itemValue="000000" />
							<f:selectItem itemLabel="#FF0000" 
								itemValue="FF0000" />
							<f:ajax render="url" />
						</h:selectOneMenu>
						<h:outputLabel value=
							"#{msg['editDonationForm.form_url']}:">
						</h:outputLabel>
						<h:inputTextarea id="url" rows="6" 
							cols="30" readonly="true"
							value="#{editDonationFormController.url}" />
					</h:panelGrid>
					<h:commandButton value="OK"
						action="#{editDonationFormController.doOk}" />
				</h:form>
			</ui:define>
		</ui:composition>
	</f:view>
</body>
</html>

Listing 4-25     Facelet editDonationForm.xhtml

The Facelet presents data in tabular form using the <h:panelGrid> tag. The table has two columns. The first column contains the names of the fields (tag <h:outputText>), while the second contains the input components. For the colors, a combo box is used as a choice component (tag <h:selectOneMenu>). The URL is displayed in a text area (tag <h:textArea>) that cannot be changed manually (attribute readonly="true"). As explained in section ‎4.6.6, we use the <f:ajax> tag within the color choice component to automatically adjust the URL in the text area in the event of a color change without having to reload the whole page. At the end of the form is the confirmation button (tag <h:commandButton>) that activates the controller method doOk. Notice here that the use of a combo box for color selection is suboptimal, because the colors must be selected by means of a Hexcode. A better option would be a choice component in which a range of colors are displayed for the user to choose from. A component like this, which is often referred to as a color picker, does not exist in the JSF standard. However, the extension PrimeFaces, which we’ll look at in section ‎4.9.2, can help us in this regard.

Adjust the Existing Classes

As with the previous use case, the editing of the donation form is dependent on the selected campaign. The selected campaign is stored in the CampaignProducer bean. This means that we need to adjust the controller class ListCampaignsController. We must modify the method doEditDonationForm in such a way that the selected campaign is set in the CampaignProducer bean.

public String doEditDonationForm(Campaign campaign) {
	campaignProducer.setSelectedCampaign(campaign);
	return Pages.EDIT_DONATION_FORM;
}

Following the obligatory Maven build step, we can now try out our newly implemented use case.  ‎Fig. 4-6 shows our screen for editing the donation form. Features we see here include the combo box choice component (tag <h:selectOneMenu>), the text area (tag <h:textarea>) and a button for submitting the form (tag <h:commandButton>).  At this point, we should briefly test out the Ajax functionality. To do this, change one of the choices in a combo box. The URL in the text area should adjust automatically to reflect the new value.

 

Fig. 4-6     Screen for editing a donation form

Generating the Form URL

The controller EditDonationFormController supplies the URL using the method getUrl().

public String getUrl() {
		return "http://localhost:8080/my-aktion/donateMoney.jsf?bgColor="
				+ bgColor + "&textColor=" + textColor + "&campaignId="
				+ campaignProducer.getSelectedCampaign().getId();
	}

We can see that the URL is specified in the program code. The hard-coded address only works if the browser and server are run on the same computer. In a real-world environment, however, this is almost never the case. We must alter the method in such a way that it generates the URL automatically with the correct protocol, host, port and web context. We can read this information from the current request. To obtain it, we must descend a little into the depths of the Java EE APIs. The class representing an HTTP request in Java EE is the javax.servlet.http.HttpServletRequest class of the Servlet API. Access to the current instance can be obtained once again via the FacesContext.

The URL is generated using two methods. The private method getAppUrl determines the URL of the web application, then the public method getUrl uses this and amends the address with the Facelet name and the parameters.

private String getAppUrl() {
	HttpServletRequest req = (HttpServletRequest) FacesContext
			.getCurrentInstance().getExternalContext().getRequest();
	String scheme = req.getScheme();
	String serverName = req.getServerName();
	int serverPort = req.getServerPort();
	String contextPath = req.getContextPath();
	return scheme + "://" + serverName + ":" + serverPort + contextPath;
}

public String getUrl() {
	return getAppUrl() + "/" + Pages.DONATE_MONEY + ".jsf" + "?bgColor="
			+ bgColor + "&textColor=" + textColor + "&campaignId="
			+ campaignProducer.getSelectedCampaign().getId();
}

We obtain a reference to the current instance of the HttpServletRequest class via the current instance of the FacesContext class. This offers us a range of methods for retrieving information from the request. The object also contains all the information for the URL, and we take advantage of this for the automatic generation of the donation form URL. We supplement this information with the name of the Facelet from the Pages class and append the current color values and the ID of the campaign as GET parameters. Once we’ve done this, the URL will always adjust itself automatically to the execution environment and the deployment of the application.

4.7.4 Donating Money

The implementation of the use case Donate money differs somewhat from the other implementations. This use case is the only one that is not used by the organizer themselves, but by a random group of Internet users.  Therefore, the Facelet is not embedded in our template, but rather defines its own page. The organizer of a campaign can send the URL of the “Donate money” page to potential donors, allowing them to use it. Contained in the URL as HTTP GET parameters are the text and background color for the donation form and the ID of the campaign to be donated to.  These data must be interpreted in the Facelet and the page displayed accordingly.

Even though this case is slightly different, we still require a controller behind the Facelet to manage the Donation object and to save the donation for the campaign after it has been sent. Since the use case is initiated by the user via the URL, the controller does not afford us access to the Campaign object associated with the donation. Without the persistence layer, assignment of the campaignId to a Campaign object is only possible with great effort. We will revisit and solve this issue in Chapter 7 with the help of JPA. The controller can then read the campaign from the database to add the donation and write both back into the database. Until then, however, we will live with the fact that the donation is not saved when the Donate button is clicked, and is therefore lost.

The DonateMoneyController Controller Class

In the DonateMoneyController class from ‎Listing  4-26, we require access to the JSF API classes FacesContext and FacesMessage to be able to dispay a donation confirmation message to the user. The current FacesContext also allows us to obtain the correct ResourceBundle, which we require for displaying the message in the correct language. A FacesMessage is created for this purpose in the method doDonation and is stored in the current FacesContext. JSF uses this to display the thank you message to the user following the execution of doDonation and the rebuilding of the page.

package de.dpunkt.myaktion.controller;

import de.dpunkt.myaktion.model.Donation;
import javax.enterprise.context.SessionScoped;
import javax.faces.application.FacesMessage;
import javax.faces.context.FacesContext;
import javax.inject.Named;
import java.io.Serializable;
import java.text.MessageFormat;
import java.util.ResourceBundle;

@SessionScoped
@Named
public class DonateMoneyController implements Serializable {
	private static final long serialVersionUID = 5493038842003809106L;
	private String textColor = "000000";
	private String bgColor = "ffffff";
	private Long campaignId;
	private Donation donation;

	public DonateMoneyController() {
		this.donation = new Donation();
	}
	public Long getCampaignId() {
		return campaignId;
	}
	public void setCampaignId(Long campaignId) {
		this.campaignId = campaignId;
	}
	public Donation getDonation() {
		return donation;
	}
	public void setDonation(Donation donation) {
		this.donation = donation;
	}
	public String getTextColor() {
		return textColor;
	}
	public void setTextColor(String textColor) {
		this.textColor = textColor;
	}
	public String getBgColor() {
		return bgColor;
	}
	public void setBgColor(String bgColor) {
		this.bgColor = bgColor;
	}
	public String doDonation() {
		final FacesContext facesContext = FacesContext.getCurrentInstance();
		final ResourceBundle resourceBundle = facesContext.getApplication()
				.getResourceBundle(facesContext, "msg");
		final String msg = resourceBundle.getString("donateMoney.thank_you");
		facesContext.addMessage(null, new FacesMessage(
				FacesMessage.SEVERITY_INFO, msg, null));
		this.donation = new Donation();
		return Pages.DONATE_MONEY;
	}
}

Listing 4-26     DonateMoneyController class

The donateMoney.xhtml Facelet

Like the previous Facelets, the donateMoney.xhtml Facelet requires a number of language-dependent messages and texts. Therefore, before we look at the Facelet itself, we need to adjust our properties file. We will add the following lines of code to the file messages_en.properties:

donateMoney.donate_money=Donate money
donateMoney.name=Name
donateMoney.iban=IBAN
donateMoney.name_of_bank=Name of bank
donateMoney.donation_amount=Donation amount
donateMoney.i_want_a_receipt=Yes, I would like a donation receipt
donateMoney.donate=Donate
donateMoney.thank_you=Thank you for your donation!

In addition to the above, we will insert the following lines of code in the messages_de.properties file:

# donateMoney.xhtml
donateMoney.donate_money=Geld spenden
donateMoney.name=Name
donateMoney.iban=IBAN
donateMoney.name_of_bank=Name der Bank
donateMoney.donation_amount=Spendenbetrag
donateMoney.i_want_a_receipt=Ja, ich möchte eine Spendenquittung
donateMoney.donate=Spenden
donateMoney.thank_you=Vielen Dank für die Spende!

At the beginning of the donateMoney.xhtml Facelet in ‎Listing  4-27, the GET parameters of the HTTP request are read out, validated and bound to the properties of the controller bean. Directly afterwards, the color values supplied by the controller are used in the style attribute of the <body> HTML tag to take care of the desired color scheme for the display. After the <h1> title in each of the languages, there is a section with error messages. First come the global messages (<h:messages globalOnly="true"/>), then the <h:message> tags for validation of the view parameter. The rest of the Facelet contains components we already know:

<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml"
	xmlns:h="http://xmlns.jcp.org/jsf/html"
	xmlns:f="http://xmlns.jcp.org/jsf/core">
<h:head>
	<title>#{msg['donateMoney.donate_money']}</title>
	<meta charset="utf-8" />
	<f:metadata>
		<f:viewParam id="bgColor" name="bgColor"
			value="#{donateMoneyController.bgColor}">
			<f:validateRegex pattern="[A-Fa-f0-9]{6}" />
		</f:viewParam>
		<f:viewParam id="textColor" name="textColor"
			value="#{donateMoneyController.textColor}">
			<f:validateRegex pattern="[A-Fa-f0-9]{6}" />
		</f:viewParam>
		<f:viewParam id="campaignId" name="campaignId"
			converter="javax.faces.Long"
			value="#{donateMoneyController.campaignId}">
		</f:viewParam>
	</f:metadata>
</h:head>
<h:body style="margin:0; 
		font-family: Verdana, sans-serif; 
		font-size: 0.9em;
		background-color: ##{donateMoneyController.bgColor}; 
		color: ##{donateMoneyController.textColor}">
	<h1>#{msg['donateMoney.donate_money']}</h1>
	<h:messages globalOnly="true" />
	<h:message for="bgColor" style="color: red;" />
	<h:message for="textColor" style="color: red;" />
	<h:message for="campaignId" style="color: red;" />
	<h:form>
		<h:panelGrid columns="3">
			<h:outputLabel value="#{msg['donateMoney.name']}:">
			</h:outputLabel>
			<h:inputText id="name"
					value="#{donateMoneyController.donation.donorName}">
				<f:validateRequired />
			</h:inputText>
			<h:message for="name" style="color: red;" />
			<h:outputLabel value="#{msg['donateMoney.iban']}:">
			</h:outputLabel>
			<h:inputText id="iban"
					value="#{donateMoneyController.donation.account.iban}">
				<f:validateRequired />
			</h:inputText>
			<h:message for="iban" style="color: red;" />
			<h:outputLabel value="#{msg['donateMoney.name_of_bank']}:">
			</h:outputLabel>
			<h:inputText id="name_bank"
				value="#{donateMoneyController.donation.account.nameOfBank}">
				<f:validateRequired />
			</h:inputText>
			<h:message for="name_bank" style="color: red;" />
			<h:outputLabel value="#{msg['donateMoney.donation_amount']}:">
			</h:outputLabel>
			<h:panelGroup>
				<h:inputText id="donationAmount"
						value="#{donateMoneyController.donation.amount}">
					<f:validateRequired />
					<f:convertNumber maxFractionDigits="2" 
						minFractionDigits="2" />
				</h:inputText> EUR
			</h:panelGroup>
			<h:message for="donationAmount" style="color: red;" />
		</h:panelGrid>
		<h:selectBooleanCheckbox
			value="#{donateMoneyController.donation.receiptRequested}" /> 
			#{msg['donateMoney.i_want_a_receipt']}<br />
		<h:commandButton value="#{msg['donateMoney.donate']}"
			action="#{donateMoneyController.doDonation}" />
	</h:form>
</h:body>
</html>

Listing 4-27     donateMoney.xhtml Facelet

We do not have to make any adjustments to the already-programmed Facelets and classes. Again, we can try out the newly implemented use case directly after the obligatory Maven build. In order to obtain a donation form for a campaign in the browser, we must enter the URL from the form Edit donation form (see ‎Fig. 4-6 ) into the browser. As an example,Fig. 4-7 shows the form for the first campaign on the list. We have already filled out the form and clicked the button Donate.

 

Fig. 4-7     Filled-out donation form

The page is reloaded and now displays the message generated by the controller (FacesMessage) underneath the title Donate money (see Fig. 4-8).

 

Fig. 4-8     Response page with the FacesMessage after submitting a completed donation form