These changes were made with the PassThruAttributeWriter and the LocalEffectEncoder, so they just pertain to ResponseWriter components, and have yet to be adapted to DOM rendering components.
The basic idea is that the javascript handler pass-through attribute rendering becomes the responsibility of the LocalEffectEncoder, not PassThruAttributeWriter anymore. That way, only one method is trying to render out the on* javascript handlers, so they're not being redundantly or incorrectly stomped over by other code.
Let's look at ice:inputText, and its attributes, as an example, dividing them up into categories.
PassThruAttributes, as defined in ExtendedAttributeConstants
Regular passthrough attributes:
accesskey
alt
dir
lang
maxlength
size
style
tabindex
title
Javascript handlers (All of these can pass-through values from the application. Renderer ones also have values from the component renderers):
Effect related handlers:
onclick
ondblclick
onkeydown
onkeypress (Renderer)
onkeyup
onmousedown
onmousemove
onmouseout
onmouseover
onmouseup
Non-Effect
onblur (Renderer)
onfocus (Renderer)
onchange
onselect
ExtendedAttributeConstants just gives a flat String[] of all those attributes. It's a little easier if we start thinking of them as sets that describe categories of attributes.
Passthru : All of the pass-through attributes for a given component
Regular : A component's non-javascript attributes
Javascript : A component's javascript on* attributes
Effect : Subset of Javascript, related to effects. Eg: onclick is in it, onblur is not.
NonEffect : Remainder of Javascript
AllPossibleJavascript : For all components together, the complete list of all possible javascript on* attributes
Before, the PassThruAttributeWriter wrote out all Passthru, both Regular and Javascript. Then LocalEffectEncoder wrote out the Effect attributes. Since Effect is a subset of Javascript, they would overlap. Now, we want PassThruAttributeWriter to only do Regular, and LocalEffectEncoder to do Javascript.
We get Javascript by finding the intersection of AllPossibleJavascript and Passthru:
private static final String[] jsEvents = LocalEffectEncoder.maskEvents(
ExtendedAttributeConstants.getAttributes(
ExtendedAttributeConstants.ICE_INPUTTEXT));
We get Regular by taking Passthru, and excluding Javascript:
private static final String[] passThruAttributes =
ExtendedAttributeConstants.getAttributesExceptJavascript(
ExtendedAttributeConstants.ICE_INPUTTEXT,
jsEvents);
Note that we have to pass our renderer values into LocalEffectEncoder, for any entries in Javascript. But, Javascript does not equal AllPossibleJavascript. So, for any entries outside of that, the component renderer has to handle them itself.
In this case, the output is highly static, only depending on if partialSubmit is true or false, so I build two static maps, taking care to use String constants, so they'll all be interned:
private static Map rendererJavascript;
private static Map rendererJavascriptPartialSubmit;
static {
rendererJavascript = new HashMap();
rendererJavascript.put(HTML.ONKEYPRESS_ATTR,
DomBasicRenderer.ICESUBMIT);
rendererJavascript.put(HTML.ONFOCUS_ATTR,
"setFocus(this.id);");
rendererJavascript.put(HTML.ONBLUR_ATTR,
"setFocus('');");
rendererJavascriptPartialSubmit = new HashMap();
rendererJavascriptPartialSubmit.put(HTML.ONKEYPRESS_ATTR,
DomBasicRenderer.ICESUBMIT);
rendererJavascriptPartialSubmit.put(HTML.ONFOCUS_ATTR,
"setFocus(this.id);");
rendererJavascriptPartialSubmit.put(HTML.ONBLUR_ATTR,
"setFocus('');" + DomBasicRenderer.ICESUBMITPARTIAL);
And then when rendering I just select between the two of them:
Map rendererJS = ((IceExtended) uiComponent).getPartialSubmit()
? rendererJavascriptPartialSubmit : rendererJavascript;
LocalEffectEncoder.encode(
facesContext, uiComponent, jsEvents, rendererJS, null, writer);
The new LocalEffectEncoder.encode
method iterates over the entries in Javascript, getting the:
1. on*effect value
2. on* application value
3. on* component renderer value
and concatenating them in that order.
Maybe the renderer could pass in a Map of what it wants to render to LocalEffectEncoder, where a map entry would look like "onclick" -> "iceSubmitPartial(...)". These could be statically defined, or dynamically generated if necessary.
LocalEffectEncoder could iterate over all the typical events, and combine the effect + user + framework Javascript, based on the on*effect attribute, the on* attribute, and the Map, respectively.
Also, it looks like some renderers invoke it directly, while it's also called from PassThruAttributeRenderer.renderHtmlAttributes, so we should probably make we're not double-invoking it that way too.