BalusC explains why this is difficult here: https://stackoverflow.com/a/7123153/3284943
Based on his suggestions, I created this:
in the JSF file, I use my custom converter and its defined attributes, to select the converter and dynamic attributes at runtime. This is a snippet from a dynamic PrimeFaces DataTable:
<h:outputText rendered = "#{column.dateTime}" value="#{table[column.name]}">
<f:converter converterId="runtimeConverterSelector" />
<f:attribute name="converterId" value="#{column.converterName}" />
<f:attribute name="pattern" value="#{column.converterPattern}" />
</h:outputText>
The custom converter:
@FacesConverter ("runtimeConverterSelector")
public class RuntimeConverterSelector implements Converter {
Converter delegateConverter;
@Override
public Object getAsObject(FacesContext context, UIComponent component, String value) {
retrieveAttributes(component);
return delegateConverter.getAsObject(context, component, value);
}
@Override
public String getAsString(FacesContext context, UIComponent component, Object value) {
retrieveAttributes(component);
return delegateConverter.getAsString(context, component, value);
}
private void retrieveAttributes(UIComponent component) {
Object attribute;
attribute = component.getAttributes().get("converterId");
if (attribute != null) {
String converterName = (String)attribute;
switch (converterName) {
case "javax.faces.BigDecimal":
delegateConverter = new BigDecimalConverter();
break;
case "javax.faces.BigInteger":
delegateConverter = new BigIntegerConverter();
break;
case "javax.faces.Boolean":
delegateConverter = new BooleanConverter();
break;
case "javax.faces.Byte":
delegateConverter = new ByteConverter();
break;
case "javax.faces.Character":
delegateConverter = new CharacterConverter();
break;
case "javax.faces.DateTimeConverter":
delegateConverter = new DateTimeConverter();
attribute = component.getAttributes().get("pattern");
if (attribute != null) ((DateTimeConverter)delegateConverter).setPattern((String)attribute);
attribute = component.getAttributes().get("timeZone");
if (attribute != null) ((DateTimeConverter)delegateConverter).setTimeZone(TimeZone.getTimeZone((String)attribute));
attribute = component.getAttributes().get("dateStyle");
if (attribute != null) ((DateTimeConverter)delegateConverter).setDateStyle((String)attribute);
attribute = component.getAttributes().get("timeStyle");
if (attribute != null) ((DateTimeConverter)delegateConverter).setDateStyle((String)attribute);
attribute = component.getAttributes().get("type");
if (attribute != null) ((DateTimeConverter)delegateConverter).setDateStyle((String)attribute);
break;
case "javax.faces.Double":
delegateConverter = new DoubleConverter();
break;
case "javax.faces.Enum":
delegateConverter = new EnumConverter();
break;
case "javax.faces.Float":
delegateConverter = new FloatConverter();
break;
case "javax.faces.Integer":
delegateConverter = new IntegerConverter();
break;
case "javax.faces.Long":
delegateConverter = new LongConverter();
break;
case "javax.faces.Number":
delegateConverter = new NumberConverter();
attribute = component.getAttributes().get("currencyCode");
if (attribute != null) ((NumberConverter)delegateConverter).setCurrencyCode((String)attribute);
attribute = component.getAttributes().get("currencySymbol");
if (attribute != null) ((NumberConverter)delegateConverter).setCurrencySymbol((String)attribute);
attribute = component.getAttributes().get("groupingUsed");
if (attribute != null) ((NumberConverter)delegateConverter).setGroupingUsed(Boolean.parseBoolean((String)attribute));
attribute = component.getAttributes().get("integerOnly");
if (attribute != null) ((NumberConverter)delegateConverter).setIntegerOnly(Boolean.parseBoolean((String)attribute));
attribute = component.getAttributes().get("locale");
if (attribute != null) ((NumberConverter)delegateConverter).setLocale(new Locale((String)attribute));
attribute = component.getAttributes().get("maxFractionDigits");
if (attribute != null) ((NumberConverter)delegateConverter).setMaxFractionDigits(Integer.parseInt((String)attribute));
attribute = component.getAttributes().get("maxIntegerDigits");
if (attribute != null) ((NumberConverter)delegateConverter).setMaxIntegerDigits(Integer.parseInt((String)attribute));
attribute = component.getAttributes().get("minFractionDigits");
if (attribute != null) ((NumberConverter)delegateConverter).setMinFractionDigits(Integer.parseInt((String)attribute));
attribute = component.getAttributes().get("minIntegerDigits");
if (attribute != null) ((NumberConverter)delegateConverter).setMinIntegerDigits(Integer.parseInt((String)attribute));
attribute = component.getAttributes().get("pattern");
if (attribute != null) ((NumberConverter)delegateConverter).setPattern((String)attribute);
attribute = component.getAttributes().get("type");
if (attribute != null) ((NumberConverter)delegateConverter).setType((String)attribute);
break;
case "javax.faces.Short":
delegateConverter = new ShortConverter();
break;
default:
System.err.println("ConverterId provided to runtimeConverterSelector, '" + converterName + "' is invalid. The default converter will be used");
break;
}
} else {
System.err.println("No converterId was provided to runtimeConverterSelector. The default converter will be used.");
}
}
}