package com.opentravelsoft.webapp.taglib;
import java.io.IOException;
import java.text.Collator;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.Locale;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;
import com.opentravelsoft.util.LabelValueBean;
/**
* Tag for creating multiple <select> options for displaying a list of
* country names.
*
* <p>
* <b>NOTE</b> - This tag requires a Java2 (JDK 1.2 or later) platform.
* </p>
*
* @author Jens Fischer, Matt Raible
* @version $Revision: 1.4.2.1 $ $Date: 2006-06-10 08:00:48 -0600 (Sat, 10 Jun 2006) $
*/
public class CountryTag extends TagSupport {
private static final long serialVersionUID = 3905528206810167095L;
private String name;
private String prompt;
private String scope;
private String selected;
public void setName(String name) {
this.name = name;
}
public void setPrompt(String prompt) {
this.prompt = prompt;
}
public void setDefault(String selected) {
this.selected = selected;
}
public void setToScope(String scope) {
this.scope = scope;
}
/**
* Process the start of this tag.
*
* @return int status
* @exception JspException if a JSP exception has occurred
* @see javax.servlet.jsp.tagext.Tag#doStartTag()
*/
public int doStartTag() throws JspException {
// ExpressionEvaluator eval = new ExpressionEvaluator(this, pageContext);
//
// if (selected != null) {
// selected = eval.evalString("default", selected);
// }
Locale userLocale = pageContext.getRequest().getLocale();
List countries = this.buildCountryList(userLocale);
if (scope != null) {
if (scope.equals("page")) {
pageContext.setAttribute(name, countries);
} else if (scope.equals("request")) {
pageContext.getRequest().setAttribute(name, countries);
} else if (scope.equals("session")) {
pageContext.getSession().setAttribute(name, countries);
} else if (scope.equals("application")) {
pageContext.getServletContext().setAttribute(name, countries);
} else {
throw new JspException("Attribute 'scope' must be: page, request, session or application");
}
} else {
StringBuffer sb = new StringBuffer();
sb.append("<select name=\"").append(name).append("\" id=\"").append(name).append("\" class=\"select\">\n");
if (prompt != null) {
sb.append(" <option value=\"\" selected=\"selected\">");
// sb.append(eval.evalString("prompt", prompt)).append("</option>\n");
}
for (Object country1 : countries) {
LabelValueBean country = (LabelValueBean) country1;
sb.append(" <option value=\"").append(country.getValue()).append("\"");
if ((selected != null) && selected.equals(country.getValue())) {
sb.append(" selected=\"selected\"");
}
sb.append('>').append(country.getLabel()).append("</option>\n");
}
sb.append("</select>");
try {
pageContext.getOut().write(sb.toString());
} catch (IOException io) {
throw new JspException(io);
}
}
return super.doStartTag();
}
// /**
// * Release aquired resources to enable tag reusage.
// *
// * @see javax.servlet.jsp.tagext.Tag#release()
// */
// public void release() {
// super.release();
// }
/**
* Build a List of LabelValues for all the available countries. Uses
* the two letter uppercase ISO name of the country as the value and the
* localized country name as the label.
*
* @param locale The Locale used to localize the country names.
*
* @return List of LabelValues for all available countries.
*/
@SuppressWarnings("unchecked")
protected List<LabelValueBean> buildCountryList(Locale locale) {
final Locale[] available = Locale.getAvailableLocales();
List<LabelValueBean> countries = new ArrayList<LabelValueBean>();
for (Locale anAvailable : available) {
final String iso = anAvailable.getCountry();
final String name = anAvailable.getDisplayCountry(locale);
if (!"".equals(iso) && !"".equals(name)) {
LabelValueBean country = new LabelValueBean(name, iso);
if (!countries.contains(country)) {
countries.add(new LabelValueBean(name, iso));
}
}
}
Collections.sort(countries, new LabelValueComparator(locale));
return countries;
}
/**
* Class to compare LabelValues using their labels with
* locale-sensitive behaviour.
*/
public class LabelValueComparator implements Comparator {
private Comparator c;
/**
* Creates a new LabelValueComparator object.
*
* @param locale The Locale used for localized String comparison.
*/
public LabelValueComparator(final Locale locale) {
c = Collator.getInstance(locale);
}
/**
* Compares the localized labels of two LabelValues.
*
* @param o1 The first LabelValue to compare.
* @param o2 The second LabelValue to compare.
*
* @return The value returned by comparing the localized labels.
*/
@SuppressWarnings("unchecked")
public final int compare(Object o1, Object o2) {
LabelValueBean lhs = (LabelValueBean) o1;
LabelValueBean rhs = (LabelValueBean) o2;
return c.compare(lhs.getLabel(), rhs.getLabel());
}
}
}