Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to programmatically instantiate a composite component or a tag component?

I would like to programmatically instantiate composite or tag components.

This instantiation would be performed by a custom component, typically adding those composite or tag components as children.

The best answer I found when crawling forums is : http://www.java.net/node/701640#comment-791881 . It looks a lot like another answer I found on this forum : How to programmatically or dynamically create a composite component in JSF 2 .

While working on this question, I finally wrote code that works for composite instanciation using MyFaces (examples in link seems to be Mojarra specific). I copy it there as it took me some time to write it and hope it will help someone else :

public UIComponent instantiateComposite(String namespace, String componentName) {
    FacesContext ctx = FacesContext.getCurrentInstance();
    Resource resource = ctx.getApplication().getResourceHandler().createResource( componentName + ".xhtml", namespace );
    UIComponent cc = ctx.getApplication().createComponent( ctx, resource );
    UIPanel panel = (UIPanel) ctx.getApplication().createComponent( UIPanel.COMPONENT_TYPE );

    // set the facelet's parent
    cc.getFacets().put( UIComponent.COMPOSITE_FACET_NAME, panel );

    FaceletFactory ff = (DefaultFaceletFactory) DefaultFaceletFactory.getInstance();
    if(ff == null) {
        FaceletViewDeclarationLanguage vdl = new FaceletViewDeclarationLanguage(ctx);

        Method createCompiler = null;
        Method createFaceletFactory = null;
        try {
            createCompiler = FaceletViewDeclarationLanguage.class.getDeclaredMethod("createCompiler",FacesContext.class);
            createFaceletFactory = FaceletViewDeclarationLanguage.class.getDeclaredMethod("createFaceletFactory",FacesContext.class,org.apache.myfaces.view.facelets.compiler.Compiler.class);
            createCompiler.setAccessible(true);
            createFaceletFactory.setAccessible(true);
            org.apache.myfaces.view.facelets.compiler.Compiler compiler = (org.apache.myfaces.view.facelets.compiler.Compiler) createCompiler.invoke(vdl, ctx);
            ff = (FaceletFactory) createFaceletFactory.invoke(vdl, ctx, compiler);
        } catch (IllegalAccessException ex) {
            Logger.getLogger(SenatDataTableEntryDetail.class.getName()).log(Level.SEVERE, null, ex);
        } catch (IllegalArgumentException ex) {
            Logger.getLogger(SenatDataTableEntryDetail.class.getName()).log(Level.SEVERE, null, ex);
        } catch (InvocationTargetException ex) {
            Logger.getLogger(SenatDataTableEntryDetail.class.getName()).log(Level.SEVERE, null, ex);
        } catch (NoSuchMethodException ex) {
            Logger.getLogger(SenatDataTableEntryDetail.class.getName()).log(Level.SEVERE, null, ex);
        } catch (SecurityException ex) {
            Logger.getLogger(SenatDataTableEntryDetail.class.getName()).log(Level.SEVERE, null, ex);
        }
    }

    try {
        Facelet facelet = ff.getFacelet(resource.getURL());
        facelet.apply( ctx, panel );
    } catch ( IOException e ) {
        e.printStackTrace();
    }
    return cc;
}

Please do not pay attention to ugly exception handling : it is just as automatically generated by netbeans... I will ask MyFaces developpers if there is a way to avoid ugly reflection hacks.

How can I do the same with tag components , I mean components declared like :

<tag>
    <description>blah blah</description>
    <tag-name>myWonderfulTag</tag-name>
    <source>tags/mwl/myWonderfulTag.xhtml</source>
<!-- attributes -->
</tag>

in taglib.

Thanks in advance.

like image 788
Ludovic Pénet Avatar asked Nov 13 '22 08:11

Ludovic Pénet


1 Answers

ok, I found how to do it with MyFaces. It also requires a bit of ugly reflection...

public static UIComponent instantiateTagComponent(Location loc, String namespace, String localPrefix, String componentName, TagComponentParam params[]) {
    final String ERROR_MESSAGE = "Erreur lors de  l'instanciation d'un tag component";
    FacesContext ctx = FacesContext.getCurrentInstance();
    AbstractFaceletContext fctx = (AbstractFaceletContext) ctx.getAttributes().get(FaceletContext.FACELET_CONTEXT_KEY);
    fctx.pushPageContext(new PageContextImpl());

    FaceletViewDeclarationLanguage vdl = new FaceletViewDeclarationLanguage(ctx);

    UIPanel panel = (UIPanel) ctx.getApplication().createComponent( UIPanel.COMPONENT_TYPE );

    try {
        Method createCompiler = FaceletViewDeclarationLanguage.class.getDeclaredMethod("createCompiler",FacesContext.class);
        Method createFaceletFactory = FaceletViewDeclarationLanguage.class.getDeclaredMethod("createFaceletFactory",FacesContext.class,org.apache.myfaces.view.facelets.compiler.Compiler.class);
        createCompiler.setAccessible(true);
        createFaceletFactory.setAccessible(true);
        org.apache.myfaces.view.facelets.compiler.Compiler compiler = (org.apache.myfaces.view.facelets.compiler.Compiler) createCompiler.invoke(vdl, ctx);
        FaceletFactory ff = (FaceletFactory) createFaceletFactory.invoke(vdl, ctx, compiler);
        TagLibrary tl = compiler.createTagLibrary();
        TagConfig tc = new JSFUtilsTagUnit(tl, namespace, localPrefix, componentName,params,loc);
        TagHandler th = tl.createTagHandler(namespace, componentName, tc);
        Field declaredField = FaceletCompositionContext.class.getDeclaredField("FACELET_COMPOSITION_CONTEXT_KEY");
        declaredField.setAccessible(true);
        FaceletCompositionContextImpl faceletCompositionContextImpl = new FaceletCompositionContextImpl(ff,ctx);
        Class<?> dfcClass = Class.forName("org.apache.myfaces.view.facelets.impl.DefaultFaceletContext");
        Field fMCTX = dfcClass.getDeclaredField("_mctx");
        fMCTX.setAccessible(true);
        fMCTX.set(fctx, faceletCompositionContextImpl);
        FacesContext.getCurrentInstance().getAttributes().put((String)declaredField.get(null),faceletCompositionContextImpl);
        FaceletCompositionContext mctx = (FaceletCompositionContext) FaceletCompositionContext.getCurrentInstance(fctx);
        mctx.startComponentUniqueIdSection();
        th.apply( fctx, panel );
    } catch (IOException ex) {
        log.error(ERROR_MESSAGE, ex);
    } catch (InvocationTargetException ex) {
        log.error(ERROR_MESSAGE, ex);
    } catch (NoSuchMethodException ex) {
        log.error(ERROR_MESSAGE, ex);
    } catch (NoSuchFieldException ex) {
        log.error(ERROR_MESSAGE, ex);
    } catch (SecurityException ex) {
        log.error(ERROR_MESSAGE, ex);
    } catch (ClassNotFoundException ex) {
        log.error(ERROR_MESSAGE, ex);
    } catch (IllegalArgumentException ex) {
        log.error(ERROR_MESSAGE, ex);
    } catch (IllegalAccessException ex) {
        log.error(ERROR_MESSAGE, ex);
    }
    finally {
        fctx.popPageContext();
    }
    return panel;
}

The JSFUtilsTagUnit class is :

static class JSFUtilsTagUnit implements TagConfig {

    private final TagLibrary library;
    private final String namespace;
    private final String name;
    private Tag tag;

    public JSFUtilsTagUnit(TagLibrary library, String namespace, String localPrefix, String name, TagComponentParam params[], Location loc) {
        this.library = library;
        this.namespace = namespace;
        this.name = name;
        if(loc == null) {
            loc = new Location("virtual", -1, -1);
        }
        TagAttribute[] tas;
        if((params != null) && (params.length > 0)) {
            tas = new TagAttribute[params.length];
            int iCurrent = 0;
            for(TagComponentParam tcpCur : params) {
                tas[iCurrent] = new TagAttributeImpl(loc,"",tcpCur.getName(), tcpCur.getName(), tcpCur.getValue());
                iCurrent++;
            }
        } else {
            tas = new TagAttribute[0];
        }
        TagAttributes tattrs = new TagAttributesImpl(tas);
        tag = new Tag(loc, namespace, name, localPrefix+":"+name, tattrs);
    }

    public FaceletHandler getNextHandler() {
        return new FaceletHandler() {

            public void apply(FaceletContext ctx, UIComponent parent) throws IOException {
            }
        };
    }

    public Tag getTag() {
        return tag;
    }

    public String getTagId() {
        return "-1";
    }

}

Hope it will help someone else !

like image 68
Ludovic Pénet Avatar answered Nov 15 '22 11:11

Ludovic Pénet