 
                        
        
                        
        org.glassfish.admingui.common.handlers.UtilHandlers Maven / Gradle / Ivy
/*
 * Copyright (c) 2024 Contributors to the Eclipse Foundation.
 * Copyright (c) 1997, 2018 Oracle and/or its affiliates. All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */
/*
 * UtilHandlers.java
 *
 * Created on August 31, 2006, 2:36 PM
 *
 * To change this template, choose Tools | Template Manager
 * and open the template in the editor.
 */
package org.glassfish.admingui.common.handlers;
import org.glassfish.admingui.common.util.GuiUtil;
import org.glassfish.admingui.common.util.JSONUtil;
import org.glassfish.admingui.common.util.RestUtil;
import com.sun.jsftemplating.annotation.Handler;
import com.sun.jsftemplating.annotation.HandlerInput;
import com.sun.jsftemplating.annotation.HandlerOutput;
import com.sun.jsftemplating.layout.LayoutDefinitionManager;
import com.sun.jsftemplating.layout.ViewRootUtil;
import com.sun.jsftemplating.layout.descriptors.LayoutElement;
import com.sun.jsftemplating.layout.descriptors.handler.HandlerContext;
import com.sun.jsftemplating.layout.descriptors.handler.HandlerDefinition;
import com.sun.jsftemplating.util.FileUtil;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.UnsupportedEncodingException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLDecoder;
import java.text.DecimalFormat;
import java.text.StringCharacterIterator;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.GregorianCalendar;
import java.util.HashMap;
import java.util.List;
import java.util.Set;
import java.util.StringTokenizer;
import java.util.Map;
import java.util.logging.Level;
import jakarta.faces.component.UIViewRoot;
/**
 *
 * @author Jennifer Chou
 */
public class UtilHandlers {
    /** Creates a new instance of UtilHandlers */
    public UtilHandlers() {
    }
    /**
     *  Adds the specified (signed) amount of time to the given calendar
     *      field, based on the calendar's rules and returns the resulting Date.
     *      See java.util.GregorianCalendar add(int field, int amount). 
     *
     *   Input value: "Field" -- Type: Integer
     *   - java.util.Calendar field
     *   Input value: "Amount" -- Type: Integer
     *   - the amount of date or time to be added to the field.
     *   Output value: "Date" -- Type: java.util.Date
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="calendarAdd",
        input={
            @HandlerInput(name="Field", type=Integer.class, required=true),
            @HandlerInput(name="Amount", type=Integer.class, required=true)},
        output={
            @HandlerOutput(name="Date", type=java.util.Date.class)})
    public static void calendarAdd(HandlerContext handlerCtx) {
        int field = ((Integer) handlerCtx.getInputValue("Field")).intValue();
        int amount = ((Integer) handlerCtx.getInputValue("Amount")).intValue();
        GregorianCalendar cal = new GregorianCalendar();
        cal.add(field, amount);
        handlerCtx.setOutputValue("Date", cal.getTime());
    }
    /**
     *  Creates a new File instance by converting the given pathname string
     *      into an abstract pathname. If the given string is the empty string,
     *      then the result is the empty abstract pathname. 
     *
     *   Input value: "Pathname" -- Type: String
     *  
 Output value: "File" -- Type: java.io.File
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="getFile",
        input={
            @HandlerInput(name="Pathname", type=String.class, required=true)},
        output={
            @HandlerOutput(name="File", type=File.class)})
    public static void getFile(HandlerContext handlerCtx) {
        String pathname = (String) handlerCtx.getInputValue("Pathname");
        handlerCtx.setOutputValue("File", pathname != null ? new File(pathname) : null);
    }
    /**
     *  This handler serves a resource via JSFTemplating's FileStreamer.
     */
    @Handler(id="gf.serveResource",
        input={
            @HandlerInput(name="path", type=String.class, required=true)},
        output={
            @HandlerOutput(name="content", type=String.class)})
    public static void serveResource(HandlerContext ctx) throws java.io.IOException {
        /*
          JSF 2.0 impl used to set the writer before the render response phase
          (in apply request values).  So we couldn't control the output of an
          Ajax request. :(  Therefor the following is commented out.  On
          10/6/2009 rlubke fixed this, we need to adjust the JS to handle a
          direct response then this can be enabled.
            FacesContext facesCtx = ctx.getFacesContext();
            UIViewRoot root = new UIViewRoot();
            root.setRenderKitId("dummy");
            facesCtx.setViewRoot(root);
            LayoutViewHandler.serveResource(facesCtx,
                (String) ctx.getInputValue("path"));
         */
        String path = (String) ctx.getInputValue("path");
        int idx = path.lastIndexOf("://");
        String port = null;
        if (idx != -1) {
            // Strip off protocol
            path = path.substring(idx + 3);
            // Now looks like: host.domain:port/resource
// FIXME: port 80 may be omitted (or 443 for https)
            if ((idx = path.indexOf(':')) != -1) {
                path = path.substring(idx + 1);
                if ((idx = path.indexOf('/')) != -1) {
                    port = path.substring(0, idx);
                    path = path.substring(idx);
                }
            }
        }
        URL url = FileUtil.searchForFile(path, null);
        if ((url == null) && (port != null)) {
            // Attempt to read from localhost
            path = "http://localhost:" + port + path;
            try {
                url = new URL(path);
            } catch (MalformedURLException ex) {
                url = null;
            }
        }
        String content = "";
        if (url != null) {
            try{
                content = new String(FileUtil.readFromURL(url));
            } catch (FileNotFoundException fnfe) {
                //
            }
        }
        // Set the output
        ctx.setOutputValue("content", content);
    }
    /**
     *  Returns the name of the file or directory denoted by this abstract
     *      pathname. This is just the last name in the pathname's name sequence.
     *      If the pathname's name sequence is empty, then the empty string is returned. 
     *
     *   Input value: "File" -- Type: java.io.File
     *  
 Output value: "Name" -- Type: String
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="fileGetName",
        input={
            @HandlerInput(name="File", type=File.class, required=true)},
        output={
            @HandlerOutput(name="Name", type=String.class)})
    public static void fileGetName(HandlerContext handlerCtx) {
        File file = (File) handlerCtx.getInputValue("File");
        handlerCtx.setOutputValue("Name", (file == null) ? "" : file.getName() );
    }
    /**
     *  Returns a duplicate copy of the source Map
     *
     *  
 Input value: "source" -- Type: java.util.Map
     *  
 Output value: "dest" -- Type: java.util.Map
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="mapCopy",
        input={
                @HandlerInput(name="source", type=Map.class, required=true)},
        output={
                @HandlerOutput(name="dest", type=Map.class)})
    public static void mapCopy(HandlerContext handlerCtx) {
        Map source = (Map) handlerCtx.getInputValue("source");
        handlerCtx.setOutputValue("dest",  new HashMap(source));
    }
    /**
     *  Returns the value to which the input map maps the input key. 
     *
     *   Input value: "Map" -- Type: java.util.Map
     *  
 Input value: "Key" -- Type: Object
     *  
 Output value: "Value" -- Type: Object
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="mapGet",
        input={
            @HandlerInput(name="Map", type=Map.class, required=true),
            @HandlerInput(name="Key", type=Object.class, required=true)},
        output={
            @HandlerOutput(name="Value", type=Object.class)})
    public static void mapGet(HandlerContext handlerCtx) {
        Map map = (Map) handlerCtx.getInputValue("Map");
        Object key = (Object) handlerCtx.getInputValue("Key");
        handlerCtx.setOutputValue("Value", (Object) map.get(key));
    }
    @Handler(id="mapJoin",
        input={
            @HandlerInput(name="map", type=Map.class, required=true),
            @HandlerInput(name="sep", type=String.class, defaultValue=","),
            @HandlerInput(name="skipBlankValues", type=Boolean.class, defaultValue="true")
        },
        output={
            @HandlerOutput(name="result", type=String.class)
        }
    )
    public static void mapJoin(HandlerContext handlerCtx) {
        Map map = (Map)handlerCtx.getInputValue("map");
        String sep = (String)handlerCtx.getInputValue("sep");
        Boolean skipBlankValues = (Boolean)handlerCtx.getInputValue("skipBlankValues");
        String sepHolder = "";
        assert(map != null);
        StringBuilder result = new StringBuilder();
        for (Map.Entry entry : (Set)map.entrySet()) {
            Object value = entry.getValue();
            if (skipBlankValues && ((value == null) || (value.toString().isEmpty()))) {
                continue;
            }
            result.append(sepHolder).append(entry.getKey()).append("=").append(entry.getValue());
            sepHolder = sep;
        }
        handlerCtx.setOutputValue("result", result.toString());
    }
    /**
     * This handler goes through all the Map entries, if the value is null, it will convert that to "false"
     * This is used usually to take care of un-checked checkbox which is set to 'null',  but needs to be set to false when
     * the map is passed in as attrsMap for request.
     *  Returns the map  
     *
     *   Input value: "map" -- Type: java.util.Map
     * 
 Input value: "key" -- Type: java.util.List
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="gf.mapValueNullToFalse",
        input={
            @HandlerInput(name="map", type=Map.class, required=true),
            @HandlerInput(name="keys", type=List.class, required=true) },
        output={
            @HandlerOutput(name="result", type=Map.class)} )
    public static void mapValueNullToFalse(HandlerContext handlerCtx) {
        Map map = (Map)handlerCtx.getInputValue("map");
        List  keys = (List)handlerCtx.getInputValue("keys");
        Map result = new HashMap(map);
        for( String key: keys){
            if (map.get(key) == null){
                result.put(key, "false");
            }
        }
        handlerCtx.setOutputValue("result", result);
    }
    /**
     *  Returns the keys  
     *
     *   Input value: "Map" -- Type: java.util.Map
     *  
 Output value: "Keys" -- Type: Object
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="gf.getMapKeys",
        input={
            @HandlerInput(name="Map", type=Map.class, required=true)},
        output={
            @HandlerOutput(name="Keys", type=List.class)})
    public static void getMapKeys(HandlerContext handlerCtx) {
        Map map = (Map) handlerCtx.getInputValue("Map");
        List keyList = new ArrayList();
        if (map == null) {
            map = new HashMap();
        }
        keyList.addAll(map.keySet());
        Collections.sort(keyList);
        handlerCtx.setOutputValue("Keys", keyList);
    }
    /**
     *  Adds the given value to a List
     *  Input list: "list" -- Type: java.util.List
     * 
 Input value: "value" -- Type: java.lang.Object
     * 
 Input value: "index" -- Type: Integer
     *
     * @param handlerCtx The HandlerContext
     */
    @Handler(id="listAdd",
        input={
            @HandlerInput(name="list", type=List.class),
            @HandlerInput(name="value", type=Object.class, required=true),
            @HandlerInput(name="index", type=Integer.class),
            @HandlerInput(name="sort", type=boolean.class, defaultValue="false"),
        },
        output={
            @HandlerOutput(name="result", type=List.class)}
    )
    public static void listAdd(HandlerContext handlerCtx) {
        List list = (List)handlerCtx.getInputValue("list");
        Integer index = (Integer)handlerCtx.getInputValue("index");
        if(list == null) {
            list = new ArrayList();
        }
        if (index == null){
            list.add(handlerCtx.getInputValue("value"));
        } else {
            list.add(index, handlerCtx.getInputValue("value"));
        }
        boolean sort = (Boolean) handlerCtx.getInputValue("sort");
        if (sort){
            Collections.sort(list);
        }
        handlerCtx.setOutputValue("result", list);
    }
     /**
     * 
 Remove the given value from a List
 if it is present
     *  Input list: "list" -- Type: java.util.List
     * 
 Input value: "value" -- Type: java.lang.Object
     *
     * @param handlerCtx The HandlerContext
     */
    @Handler(id="listRemove",
        input={
            @HandlerInput(name="list", type=List.class),
            @HandlerInput(name="value", type=Object.class, required=true)
        },
        output={
            @HandlerOutput(name="result", type=List.class)}
    )
    public static void listRemove(HandlerContext handlerCtx) {
        List list = (List)handlerCtx.getInputValue("list");
        if(list == null) {
            list = new ArrayList();
        }
        list.remove(handlerCtx.getInputValue("value"));
        handlerCtx.setOutputValue("result", list);
    }
    /**
     * 
 sort a List
     *  Input list: "list" -- Type: java.util.List
     *
     * @param handlerCtx The HandlerContext
     */
    @Handler(id="gf.listSort",
        input={
            @HandlerInput(name="list", type=List.class, required=true)}
    )
    public static void listSort(HandlerContext handlerCtx) {
        List list = (List)handlerCtx.getInputValue("list");
        Collections.sort(list);
    }
    /**
     * 
 Combine 2 lists List by adding the object in the 2nd list to the first list
     *  Input value: "list" -- Type: java.util.List
     * 
 Input value: "list2" -- Type: java.util.List
     *
     * @param handlerCtx The HandlerContext
     */
    @Handler(id="gf.listCombine",
        input={
            @HandlerInput(name="list", type=List.class, required=true),
            @HandlerInput(name="list2", type=List.class, required=true)
        },
        output={
            @HandlerOutput(name="result", type=List.class)}
    )
    public static void listCombine(HandlerContext handlerCtx) {
        List list = (List)handlerCtx.getInputValue("list");
        List list2 = (List)handlerCtx.getInputValue("list2");
        if (list == null) {
            list = new ArrayList();
        }
        if (list2 != null) {
            for(Object one : list2) {
                    list.add(one);
            }
        }
        handlerCtx.setOutputValue("result", list);
    }
    /**
     * 
 Test if a list Listcontains the string 
     *  Input value: "list" -- Type: java.util.List
     * 
 Input value: "testStr" -- Type: String
     * 
 Output value: "contain" -- Type: Boolean
     */
    @Handler(id="gf.containedIn",
        input={
            @HandlerInput(name="list", type=List.class, required=true),
            @HandlerInput(name="testStr", type=String.class, required=true)},
        output={
            @HandlerOutput(name="contain", type=Boolean.class)})
    public static void containedIn(HandlerContext handlerCtx) {
        List list = (List)handlerCtx.getInputValue("list");
        boolean contain = (list==null) ? false : list.contains(handlerCtx.getInputValue("testStr"));
        handlerCtx.setOutputValue("contain",  contain);
    }
    /**
     * 
 Compare if 2 objects is equal 
     *
     *   Input value: "obj1" -- Type: Object
     *  
 Input value: "obj2" -- Type: Object
     *  
 Output value: "equal" -- Type: Object
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="compare",
        input={
            @HandlerInput(name="obj1", type=Object.class, required=true),
            @HandlerInput(name="obj2", type=Object.class, required=true)},
        output={
            @HandlerOutput(name="objEqual", type=Boolean.class)})
    public static void compare(HandlerContext handlerCtx) {
        boolean ret = false;
        Object obj1 = (Object) handlerCtx.getInputValue("obj1");
        Object obj2 = (Object) handlerCtx.getInputValue("obj2");
        if (obj1 != null){
            ret = obj1.equals(obj2);
        }else{
            if (obj2 == null)
                ret = true;
        }
        handlerCtx.setOutputValue("objEqual", ret);
    }
    /**
     *  This method displays the save successful message when the page refresh.
     * @param handlerCtx The HandlerContext.
     */
   @Handler(id="prepareSuccessfulMsg")
    public static void prepareSuccessful(HandlerContext handlerCtx){
        GuiUtil.prepareSuccessful(handlerCtx);
    }
    /**
     * 
 This method sets the attributes that will be used by the alert component
     *     display the message to user.
     *     If type is not specifed, it will be 'info' by default.
     * 
 Input value: "summary" -- Type: java.lang.String
     *  Input value: "type" -- Type: java.lang.String
     *  Input value: "detail" -- Type: java.lang.String
     * @param handlerCtx The HandlerContext.
     */
     @Handler(id="prepareAlertMsg",
         input={
            @HandlerInput(name="summary", type=String.class, required=true),
            @HandlerInput(name="type",  type=String.class),
            @HandlerInput(name="detail",  type=String.class)
          })
    public static void prepareAlertMsg(HandlerContext handlerCtx){
        String summary = (String) handlerCtx.getInputValue("summary");
        String type = (String) handlerCtx.getInputValue("type");
        String detail = (String) handlerCtx.getInputValue("detail");
        GuiUtil.prepareAlert(type, summary, detail);
    }
    /**
     *  This handler will test if a String starts with another String.
     */
    @Handler(id="startsWith",
        input={
            @HandlerInput(name="testStr", type=String.class, required=true),
            @HandlerInput(name="pattern", type=String.class, required=true)},
        output={
            @HandlerOutput(name="result", type=Boolean.class)})
    public static void startsWith(HandlerContext handlerCtx) {
        String testStr = ((String) handlerCtx.getInputValue("testStr"));
        String pattern = ((String) handlerCtx.getInputValue("pattern"));
        handlerCtx.setOutputValue("result", testStr.startsWith(pattern));
    }
    /**
     *  This method decodes a String using "UTF-8" as default
     * if scheme is not specified.
     */
     @Handler(id="decodeString",
         input={
            @HandlerInput(name="str", type=String.class, required=true),
            @HandlerInput(name="scheme", type=String.class)},
         output={
            @HandlerOutput(name="output", type=String.class)
        })
    public static void decodeString(HandlerContext handlerCtx) {
        String str = (String) handlerCtx.getInputValue("str");
        String scheme = (String) handlerCtx.getInputValue("scheme");
        if (GuiUtil.isEmpty(str)){
            handlerCtx.setOutputValue("output", "");
            return;
        }
        if (GuiUtil.isEmpty(scheme))
            scheme = "UTF-8";
        try{
            String output=URLDecoder.decode(str, scheme);
            handlerCtx.setOutputValue("output", output);
        }catch(UnsupportedEncodingException ex) {
            handlerCtx.setOutputValue("output", str);
            GuiUtil.getLogger().info(GuiUtil.getCommonMessage("log.error.decodeString") + ex.getLocalizedMessage());
            if (GuiUtil.getLogger().isLoggable(Level.FINE)){
                ex.printStackTrace();
            }
        }
     }
    @Handler(id="roundTo2DecimalPoint",
        input={
            @HandlerInput(name="input", type=Double.class)},
        output={
            @HandlerOutput(name="output", type=String.class)
        })
    public static void roundTo2DecimalPoint(HandlerContext handlerCtx) {
        DecimalFormat df= new DecimalFormat();
        df.setMaximumFractionDigits(2);
        try{
            Double input = (Double) handlerCtx.getInputValue("input");
            String output = (input==null)? "": df.format(input);
            handlerCtx.setOutputValue("output", output);
        }catch (Exception ex){
            handlerCtx.setOutputValue("output", "");
            GuiUtil.getLogger().info(GuiUtil.getCommonMessage("log.error.roundTo2DecimalPoint") + ex.getLocalizedMessage());
            if (GuiUtil.getLogger().isLoggable(Level.FINE)){
                ex.printStackTrace();
            }
        }
    }
    /*
     * Add an empty string as the first element to the list.
     * This is useful as the labels/values of a dropdown list, where user is not required
     * to select a value in the list.  eg. virtualServer in the deployment screen,
     * defaultWebModule in the server etc.
     */
    @Handler(id = "addEmptyFirstElement",
        input = {
            @HandlerInput(name = "in", type = List.class, required = true)},
        output = {
            @HandlerOutput(name = "out", type = List.class)
        })
    public static void addEmptyFirstElement(HandlerContext handlerCtx) {
        List in = (List) handlerCtx.getInputValue("in");
        ArrayList ar = null;
        if (in == null){
             ar = new ArrayList();
        }else{
             ar = new ArrayList(in);
        }
        ar.add(0, "");
        handlerCtx.setOutputValue("out", ar);
    }
    @Handler(id = "getListBoxOptions",
        input = {
            @HandlerInput(name = "availableList", type = List.class, required = true),
            @HandlerInput(name = "selectedCommaString", type = String.class, required = true),
            @HandlerInput(name = "addEmptyFirstChoice", type = String.class, defaultValue = "true")},
        output = {
            @HandlerOutput(name = "availableListResult", type = List.class),
            @HandlerOutput(name = "selectedOptions", type = String[].class)
        })
    public static void getListBoxOptions(HandlerContext handlerCtx) {
        String selectedCommaString = (String) handlerCtx.getInputValue("selectedCommaString");
        List availableList = (List) handlerCtx.getInputValue("availableList");
        String addEmptyFirstChoice = (String) handlerCtx.getInputValue("addEmptyFirstChoice");
        String[] selectedOptions = null;
        if ("true".equals(addEmptyFirstChoice)){
            if (availableList == null){
                availableList = new ArrayList();
            }
            availableList.add(0, "");
        }
        if (availableList != null && (availableList.size() > 0) ) {
            selectedOptions = GuiUtil.stringToArray(selectedCommaString, ",");
            if ( !(selectedOptions.length > 0)) {
                //None is selected by default
                selectedOptions = new String[]{availableList.get(0)};
            }
        }
        handlerCtx.setOutputValue("availableListResult", availableList);
        handlerCtx.setOutputValue("selectedOptions", selectedOptions);
    }
    @Handler(id = "convertArrayToCommaString",
        input = {
            @HandlerInput(name = "array", type = String[].class, required = true)},
        output = {
            @HandlerOutput(name = "commaString", type = String.class)})
    public static void convertArrayToCommaString(HandlerContext handlerCtx) {
        String[] array = (String[])handlerCtx.getInputValue("array");
        String commaString = "";
        if( (array != null) && array.length > 0 ) {
                commaString = GuiUtil.arrayToString(array, ",");
        }
        handlerCtx.setOutputValue("commaString", commaString);
    }
    @Handler(id = "convertListToCommaString",
        input = {
            @HandlerInput(name = "list", type = List.class, required = true)},
        output = {
            @HandlerOutput(name = "commaString", type = String.class)})
    public static void convertListToCommaString(HandlerContext handlerCtx) {
        List list = (List)handlerCtx.getInputValue("list");
        String commaString = "";
                if( (list != null) && list.size() > 0 ) {
                        commaString = GuiUtil.listToString(list, ",");
                }
        handlerCtx.setOutputValue("commaString", commaString);
    }
    @Handler(id = "gf.resolveTokens",
        input = {
            @HandlerInput(name = "tokens", type = List.class, required = true),
            @HandlerInput(name = "endPoint", type = String.class, required = true)},
        output = {
            @HandlerOutput(name = "resolvedTokens", type = List.class)})
    public static void resolveTokens(HandlerContext handlerCtx) {
        List tokens = (List)handlerCtx.getInputValue("tokens");
        ArrayList resolvedTokens = new ArrayList();
        String endPoint = (String)handlerCtx.getInputValue("endPoint");
        for (String token : tokens)
            resolvedTokens.add(RestUtil.resolveToken(endPoint, token));
        handlerCtx.setOutputValue("resolvedTokens", resolvedTokens);
    }
    @Handler(id = "convertListToArray",
        input = {
            @HandlerInput(name = "list", type = List.class, required = true)},
        output = {
            @HandlerOutput(name = "array", type = String[].class)})
    public static void convertListToArray(HandlerContext handlerCtx) {
        List list = (List)handlerCtx.getInputValue("list");
        handlerCtx.setOutputValue("array", list.toArray(new String[list.size()]) );
    }
   /**
     *  This handler takes in a string with delimiter and returns list
     */
    @Handler(id="convertStringtoList",
         input={
            @HandlerInput(name="str", type=String.class),
            @HandlerInput(name="delimiter", type=String.class, defaultValue=",")
            },
        output = {
            @HandlerOutput(name = "result", type = List.class)
            })
    public static void convertStringtoListHandler(HandlerContext handlerCtx) {
        List result = convertStringToList((String) handlerCtx.getInputValue("str"),
                (String) handlerCtx.getInputValue("delimiter"));
        handlerCtx.setOutputValue("result", result);
    }
    private static List convertStringToList(String str, String delimiter) {
        List result = new ArrayList();
        if (str != null) {
            if (delimiter == null) {
                delimiter = ",";
            }
            StringTokenizer tokens = new StringTokenizer(str, delimiter);
            while (tokens.hasMoreTokens()) {
                String token = tokens.nextToken().trim();
                result.add(token);
            }
        }
        return result;
    }
    @Handler(id="convertStringToMap",
        input={
            @HandlerInput(name="str", type=String.class),
            @HandlerInput(name="delimiter", type=String.class)
            },
        output = {
            @HandlerOutput(name = "result", type = Map.class)
            })
    public static void convertStringToMap(HandlerContext handlerCtx) {
        Map output = new HashMap();
        List list = convertStringToList((String) handlerCtx.getInputValue("str"),
                (String) handlerCtx.getInputValue("delimiter"));
        for (String item : list) {
            String[] parts = item.split("=");
            String key = parts[0];
            String value = "";
            if (parts.length > 1) {
                value = parts[1];
            }
            output.put(key, value);
        }
        handlerCtx.setOutputValue("result", output);
    }
     //This is the reserve of the above method.
    //We want to separator and display each jar in one line in the text box.
    @Handler(id = "formatStringsforDisplay",
        input = {
            @HandlerInput(name = "string", type = String.class, required = true)},
        output = {
            @HandlerOutput(name = "formattedString", type = String.class)})
    public static void formatStringsforDisplay(HandlerContext handlerCtx) {
        String values = (String) handlerCtx.getInputValue("string");
        if (values == null || GuiUtil.isEmpty(values.trim())) {
            handlerCtx.setOutputValue("formattedString", "");
        } else {
            String s1 = values.trim().replaceAll("\\.jar:", "\\.jar\\$\\{path.separator\\}");
            String s2 = s1.replaceAll("\\.jar;", "\\.jar\\$\\{path.separator\\}");
            String[] strArray = s2.split("\\$\\{path.separator\\}");
            StringBuilder result = new StringBuilder("");
            for (String s : strArray) {
                result.append(s).append("\n");
            }
            handlerCtx.setOutputValue("formattedString", result.toString().trim());
        }
    }
    //This converts any tab/NL etc to ${path.separator} before passing to backend for setting.
    //In domain.xml, it will be written out like  c:foo.jar${path.separator}c:bar.jar
    @Handler(id = "formatPathSeperatorStringsforSaving",
        input = {
            @HandlerInput(name = "string", type = String.class, required = true)},
        output = {
            @HandlerOutput(name = "formattedString", type = String.class)})
    public static void formatPathSeperatorStringsforSaving(HandlerContext handlerCtx) {
        String values = (String) handlerCtx.getInputValue("string");
        StringBuilder token = new StringBuilder("");
        String sep = "";
        if ((values != null) &&
                (values.trim().length() != 0)) {
            List strList = GuiUtil.parseStringList(values, "\t\n\r\f");
            for(String nextToken : strList){
                token.append(sep).append(nextToken);
                sep = PATH_SEPARATOR;
            }
        }
        handlerCtx.setOutputValue("formattedString", token.toString());
    }
    /**
     *
     */
    @Handler(id="addHandler",
        input={
            @HandlerInput(name="id", type=String.class, required=true),
            @HandlerInput(name="desc", type=String.class),
            @HandlerInput(name="class", type=String.class, required=true),
            @HandlerInput(name="method", type=String.class, required=true)
        })
    public static void addHandler(HandlerContext handlerCtx) {
        String id = (String) handlerCtx.getInputValue("id");
        String desc = (String) handlerCtx.getInputValue("desc");
        String cls = (String) handlerCtx.getInputValue("class");
        String meth = (String) handlerCtx.getInputValue("method");
        HandlerDefinition def = new HandlerDefinition(id);
        def.setHandlerMethod(cls, meth);
        if (desc != null) {
            def.setDescription(desc);
        }
        LayoutDefinitionManager.addGlobalHandlerDefinition(def);
    }
    /**
     *  A utility handler that resembles the for() method in Java.  Handler inside the for loop will be executed
     *  in a loop.  start index is specified by "start",  till less than "end".
     * eg. forLoop(start="1"  end="3" varName="foo"){}, handler inside the {} will be executed 2 times.
     *
     *  
 Input value: "start" -- Type: Integer Start index, default to Zero is not specified
     *  
 Input value: "end" -- Type: Integer End index.
     *  
 Input value: "varName" -- Type: String  Variable to be replaced in the for loop by the index.
     * @param        handlerCtx        The HandlerContext.
     */
    @Handler(id="forLoop",
        input={
            @HandlerInput(name="start", type=String.class),
            @HandlerInput(name="end", type=Integer.class, required=true),
            @HandlerInput(name="varName", type=String.class, required=true)}
        )
    public static boolean forLoop(HandlerContext handlerCtx) {
        String startInt = (String) handlerCtx.getInputValue("start");
        int start = (startInt == null) ? 0 : Integer.parseInt(startInt);
        int end = ((Integer) handlerCtx.getInputValue("end")).intValue();
        String varName = ((String) handlerCtx.getInputValue("varName"));
        List handlers = handlerCtx.getHandler().getChildHandlers();
                if (handlers.size() > 0) {
            LayoutElement elt = handlerCtx.getLayoutElement();
            Map requestMap = handlerCtx.getFacesContext().getExternalContext().getRequestMap();
            for(int ix=start;  ix<=end; ix++){
                requestMap.put(varName, ix);
                //ignore whats returned by the handler.
                elt.dispatchHandlers(handlerCtx, handlers);
                    }
                }
        return false;
    }
    /**
     *  This handler provides the foreach loop functionality.  You should
     *     specify a request attribute 'var' that will be used as the key for
     *     storing each token in the list.  You can then retreive each value
     *     as the loop iterates by requesting the request scoped attribute
     *     keyed by the value you suplied for 'var'.  You must also specify
     *     the List<Object> to iterate over.
     */
    @Handler(id="foreach",
        input={
            @HandlerInput(name="var", type=String.class, required=false, defaultValue="idx"),
            @HandlerInput(name="list", type=Collection.class, required=true) })
    public static boolean foreach(HandlerContext handlerCtx) {
        String var = (String) handlerCtx.getInputValue("var");
        Collection