org.carewebframework.ui.zk.RowComparator.java Source code

Java tutorial

Introduction

Here is the source code for org.carewebframework.ui.zk.RowComparator.java

Source

/**
 * This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. 
 * If a copy of the MPL was not distributed with this file, You can obtain one at 
 * http://mozilla.org/MPL/2.0/.
 * 
 * This Source Code Form is also subject to the terms of the Health-Related Additional
 * Disclaimer of Warranty and Limitation of Liability available at
 * http://www.carewebframework.org/licensing/disclaimer.
 */
package org.carewebframework.ui.zk;

import java.io.Serializable;
import java.lang.reflect.Method;
import java.util.Comparator;
import java.util.List;

import org.apache.commons.lang.StringUtils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import org.zkoss.zk.ui.Component;
import org.zkoss.zul.Column;
import org.zkoss.zul.Listheader;

/**
 * Serves as a generic comparator for list and grid displays.
 */
public class RowComparator implements Comparator<Object>, Serializable {

    private static final Log log = LogFactory.getLog(RowComparator.class);

    private static final long serialVersionUID = 1L;

    private final boolean _asc;

    private final String _beanProperty;

    /**
     * Automatically wires column or list headers to generic comparators. This is done by deriving
     * the getter method for each header element from the element's id. This is done by prepending
     * "get" to the header id and using that method name as the getter when comparing values across
     * rows under that header. If no id is specified for a header, no comparator is generated for
     * that header. ZK-generated id's are excluded.
     * 
     * @param headers List of headers (of type Column or Listheader).
     */
    public static void autowireColumnComparators(List<Component> headers) {
        for (Component cmp : headers) {
            if (cmp instanceof Column) {
                Column col = (Column) cmp;
                String getter = getterMethod(col);

                if (getter != null) {
                    col.setSortAscending(new RowComparator(true, getter));
                    col.setSortDescending(new RowComparator(false, getter));
                }
            } else if (cmp instanceof Listheader) {
                Listheader hdr = (Listheader) cmp;
                String getter = getterMethod(hdr);

                if (getter != null) {
                    hdr.setSortAscending(new RowComparator(true, getter));
                    hdr.setSortDescending(new RowComparator(false, getter));
                }
            }
        }
    }

    /**
     * Constructs a row comparator.
     * 
     * @param asc If true, an ascending comparator is created. If false, a descending comparator is
     *            created.
     * @param beanProperty This is the name of the getter method that will be used to retrieve a
     *            value from the underlying model object for comparison.
     */
    public RowComparator(boolean asc, String beanProperty) {
        _asc = asc;
        _beanProperty = beanProperty;
    }

    /**
     * Performs a comparison between two objects. If the objects implement the Comparable interface,
     * that method is used. Otherwise, the objects are converted to their string representations and
     * that method is used. Null values are handled.
     */
    @SuppressWarnings("unchecked")
    @Override
    public int compare(Object o1, Object o2) {
        Object v1 = getValue(o1), v2 = getValue(o2);
        int result;

        if (v1 == null && v2 == null) {
            result = 0;
        } else if (v2 == null) {
            result = -1;
        } else if (v1 == null) {
            result = 1;
        } else if (v1 instanceof Comparable && !(v1 instanceof String)) {
            result = ((Comparable<Object>) v1).compareTo(v2);
        } else {
            result = v1.toString().compareToIgnoreCase(v2.toString());
        }

        return _asc ? result : -result;
    }

    /**
     * Gets a value from the model object using the getter method specified in _beanProperty.
     * 
     * @param o The model object.
     * @return Value returned by the getter method.
     */
    private Object getValue(Object o) {
        try {
            Object[] params = null;
            Method method = o.getClass().getMethod(_beanProperty, (Class<?>[]) params);
            return method.invoke(o, params);
        } catch (Exception e) {
            log.error(e.getMessage(), e);
            return null;
        }
    }

    /**
     * Derives the name of the getter method from the component id, following "JavaBean"
     * conventions. Supports traditional property "getters" as well as boolean getter methods (i.e.
     * isActive(), hasChildren()). If the id is mapped to a boolean getter, then your id should
     * begin with 'is' or 'has'.
     * 
     * @param component Component used to derive getter method.
     * @return Null if the component has no id or has a ZK-generated id. Otherwise, if the id is
     *         prefixed with a standard getter prefix ("get", "has", "is"), it is assumed to be the
     *         name of the getter method. Lacking such a prefix, a prefix of "get" is prepended to
     *         the case-adjusted id to obtain the getter method.
     */
    private static String getterMethod(Component component) {
        String id = component.getId();

        if (id == null || id.isEmpty() || id.startsWith("z")) {
            return null;
        }

        String lc = id.toLowerCase();

        if (lc.startsWith("is") || lc.startsWith("has") || lc.startsWith("get")) {
            return StringUtils.uncapitalize(id);//i.e. isActive, hasChildren
        }
        return "get".concat(StringUtils.capitalize(id));//i.e. getOrderStatus, getTitle
    }

}