/*****************************************************************
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.cayenne.reflect;
import java.util.Collection;
import org.apache.cayenne.Fault;
import org.apache.cayenne.ValueHolder;
import org.apache.cayenne.util.PersistentObjectCollection;
/**
* A generic superclass of CollectionProperty implementations.
*
* @since 1.2
*/
public abstract class BaseToManyProperty extends BaseArcProperty implements
ToManyProperty {
public BaseToManyProperty(ClassDescriptor owner, ClassDescriptor targetDescriptor,
Accessor accessor, String reverseName) {
super(owner, targetDescriptor, accessor, reverseName);
}
@Override
public Object readProperty(Object object) throws PropertyException {
return ensureCollectionValueHolderSet(object);
}
/**
* Wraps list in a value holder that performs lazy faulting.
*/
@Override
public void writePropertyDirectly(Object object, Object oldValue, Object newValue)
throws PropertyException {
if (newValue instanceof Fault) {
super.writePropertyDirectly(object, null, newValue);
}
else {
// must resolve value holder...
ValueHolder holder = (ValueHolder) readProperty(object);
holder.setValueDirectly(newValue);
}
}
public void addTarget(Object source, Object target, boolean setReverse) {
if (target == null) {
throw new NullPointerException("Attempt to add null object.");
}
// TODO, Andrus, 2/9/2006 - CayenneDataObject differences:
// * invokes "willConnect"
// * has a callback to ObjectStore to handle flattened
// * has a callback to ObjectStore to retain snapshot
// * changes object state to modified
// Now do the rest of the normal handling (regardless of whether it was
// flattened or not)
Collection<Object> collection = (Collection<Object>) readProperty(source);
collection.add(target);
if (setReverse) {
setReverse(source, null, target);
}
}
public void addTargetDirectly(Object source, Object target) throws PropertyException {
((PersistentObjectCollection) readProperty(source)).addDirectly(target);
}
public void removeTargetDirectly(Object source, Object target) throws PropertyException {
((PersistentObjectCollection) readProperty(source)).removeDirectly(target);
}
public void removeTarget(Object source, Object target, boolean setReverse) {
// TODO, Andrus, 2/9/2006 - CayenneDataObject differences:
// * has a callback to ObjectStore to handle flattened
// * changes object state to modified
// Now do the rest of the normal handling (regardless of whether it was
// flattened or not)
Collection<Object> collection = (Collection<Object>) readProperty(source);
collection.remove(target);
if (target != null && setReverse) {
setReverse(source, target, null);
}
}
@Override
public boolean visit(PropertyVisitor visitor) {
return visitor.visitToMany(this);
}
/**
* Injects a List in the object if it hasn't been done yet.
*/
@Override
public void injectValueHolder(Object object) throws PropertyException {
ensureCollectionValueHolderSet(object);
}
/**
* Checks that an object's List field described by this property is set, injecting a
* List if needed.
*/
protected ValueHolder ensureCollectionValueHolderSet(Object object)
throws PropertyException {
Object value = accessor.getValue(object);
if (value == null || value instanceof Fault) {
value = createCollectionValueHolder(object);
accessor.setValue(object, value);
}
return (ValueHolder) value;
}
/**
* Creates a Collection for an object.
*/
protected abstract ValueHolder createCollectionValueHolder(Object object)
throws PropertyException;
}