/**
* L2FProd.com Common Components 7.3 License.
*
* Copyright 2005-2007 L2FProd.com
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.l2fprod.common.propertysheet;
import com.l2fprod.common.beans.BeanUtils;
import java.lang.reflect.Method;
import java.util.*;
/**
* DefaultProperty. <br>
*/
public class DefaultProperty extends AbstractProperty {
private String name;
private String displayName;
private String shortDescription;
private Class type;
private boolean editable = true;
private String category;
private Property parent;
private List subProperties = new ArrayList();
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getDisplayName() {
return displayName;
}
public void setDisplayName(String displayName) {
this.displayName = displayName;
}
public String getShortDescription() {
return shortDescription;
}
public void setShortDescription(String shortDescription) {
this.shortDescription = shortDescription;
}
public Class getType() {
return type;
}
public void setType(Class type) {
this.type = type;
}
public boolean isEditable() {
return editable;
}
public void setEditable(boolean editable) {
this.editable = editable;
}
public String getCategory() {
return category;
}
public void setCategory(String category) {
this.category = category;
}
/**
* Reads the value of this Property from the given object. It uses reflection
* and looks for a method starting with "is" or "get" followed by the
* capitalized Property name.
*/
public void readFromObject(Object object) {
try {
Method method = BeanUtils.getReadMethod(object.getClass(), getName());
if (method != null) {
Object value = method.invoke(object, (Object[]) null);
initializeValue(value); // avoid updating parent or firing property change
if (value != null) {
for (Iterator iter = subProperties.iterator(); iter.hasNext(); ) {
Property subProperty = (Property) iter.next();
subProperty.readFromObject(value);
}
}
}
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* Writes the value of the Property to the given object. It uses reflection
* and looks for a method starting with "set" followed by the capitalized
* Property name and with one parameter with the same type as the Property.
*/
public void writeToObject(Object object) {
try {
Method method = BeanUtils.getWriteMethod(object.getClass(), getName(), getType());
if (method != null) {
method.invoke(object, new Object[]{getValue()});
}
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/* (non-Javadoc)
* @see com.l2fprod.common.propertysheet.Property#setValue(java.lang.Object)
*/
public void setValue(Object value) {
super.setValue(value);
if (parent != null) {
Object parentValue = parent.getValue();
if (parentValue != null) {
writeToObject(parentValue);
parent.setValue(parentValue);
}
}
if (value != null) {
for (Iterator iter = subProperties.iterator(); iter.hasNext(); ) {
Property subProperty = (Property) iter.next();
subProperty.readFromObject(value);
}
}
}
public int hashCode() {
return 28 + ((name != null) ? name.hashCode() : 3) + ((displayName != null) ? displayName.hashCode() : 94) +
((shortDescription != null) ? shortDescription.hashCode() : 394) + ((category != null) ? category.hashCode() : 34) +
((type != null) ? type.hashCode() : 39) + Boolean.valueOf(editable).hashCode();
}
/**
* Compares two DefaultProperty objects. Two DefaultProperty objects are equal
* if they are the same object or if their name, display name, short
* description, category, type and editable property are the same. Note the
* property value is not considered in the implementation.
*/
public boolean equals(Object other) {
if (other == null || getClass() != other.getClass()) {
return false;
}
if (other == this) {
return true;
}
DefaultProperty dp = (DefaultProperty) other;
return compare(name, dp.name) && compare(displayName, dp.displayName) && compare(shortDescription, dp.shortDescription) &&
compare(category, dp.category) && compare(type, dp.type) && editable == dp.editable;
}
private boolean compare(Object o1, Object o2) {
return (o1 != null) ? o1.equals(o2) : o2 == null;
}
public String toString() {
return "name=" + getName() + ", displayName=" + getDisplayName() + ", type=" + getType() + ", category=" + getCategory() +
", editable=" + isEditable() + ", value=" + getValue();
}
public Property getParentProperty() {
return parent;
}
public void setParentProperty(Property parent) {
this.parent = parent;
}
public Property[] getSubProperties() {
return (Property[]) subProperties.toArray(new Property[subProperties.size()]);
}
public void clearSubProperties() {
for (Iterator iter = this.subProperties.iterator(); iter.hasNext(); ) {
Property subProp = (Property) iter.next();
if (subProp instanceof DefaultProperty)
((DefaultProperty) subProp).setParentProperty(null);
}
this.subProperties.clear();
}
public void addSubProperties(Collection subProperties) {
this.subProperties.addAll(subProperties);
for (Iterator iter = this.subProperties.iterator(); iter.hasNext(); ) {
Property subProp = (Property) iter.next();
if (subProp instanceof DefaultProperty)
((DefaultProperty) subProp).setParentProperty(this);
}
}
public void addSubProperties(Property[] subProperties) {
this.addSubProperties(Arrays.asList(subProperties));
}
public void addSubProperty(Property subProperty) {
this.subProperties.add(subProperty);
if (subProperty instanceof DefaultProperty)
((DefaultProperty) subProperty).setParentProperty(this);
}
}