blob: 460c9ef7623273f2ab565368789640ddda9d9efe [file] [log] [blame]
/*
* 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.cocoon.forms.binding;
import org.apache.avalon.framework.CascadingRuntimeException;
import org.apache.cocoon.forms.formmodel.Widget;
import org.apache.cocoon.util.ClassUtils;
import org.apache.commons.jxpath.AbstractFactory;
import org.apache.commons.jxpath.JXPathContext;
import org.apache.commons.jxpath.Pointer;
/**
* ContextJXPathBinding provides an implementation of a {@link Binding}
* that narrows the binding scope to some xpath-context on the target
* objectModel to load and save from.
*
* @version $Id$
*/
public class ContextJXPathBinding extends ComposedJXPathBindingBase {
/**
* the relative contextPath for the sub-bindings of this context
*/
private final String xpath;
/**
* The name of a factory class for building intermediate elements. Must implement
* {@link org.apache.commons.jxpath.AbstractFactory}.
*/
private AbstractFactory factory;
/**
* Constructs ContextJXPathBinding for the specified xpath sub-context
*/
public ContextJXPathBinding(JXPathBindingBuilderBase.CommonAttributes commonAtts,
String contextPath,
JXPathBindingBase[] childBindings) {
super(commonAtts, childBindings);
this.xpath = contextPath;
}
/**
* Constructs ContextJXPathBinding for the specified xpath sub-context and optional JXPath factory class.
*/
public ContextJXPathBinding(JXPathBindingBuilderBase.CommonAttributes commonAtts,
String contextPath,
String factoryClassName,
JXPathBindingBase[] childBindings) {
super(commonAtts, childBindings);
this.xpath = contextPath;
if (factoryClassName != null) {
try {
this.factory = (AbstractFactory) ClassUtils.newInstance(factoryClassName);
} catch (Exception e) {
throw new CascadingRuntimeException("Cannot create an instance of " + factoryClassName, e);
}
}
}
public String getFactoryClassName() {
return this.factory == null ? null : this.factory.getClass().getName();
}
/**
* Actively performs the binding from the ObjectModel wrapped in a jxpath
* context to the CocoonForm.
*/
public void doLoad(Widget frmModel, JXPathContext jxpc) throws BindingException {
Pointer ptr = jxpc.getPointer(this.xpath);
if (ptr.getNode() != null) {
JXPathContext subContext = jxpc.getRelativeContext(ptr);
super.doLoad(frmModel, subContext);
if (getLogger().isDebugEnabled()) {
getLogger().debug("done loading " + this);
}
} else {
if (getLogger().isDebugEnabled()) {
getLogger().debug("non-existent path: skipping " + this);
}
}
}
/**
* Actively performs the binding from the CocoonForm to the ObjectModel
* wrapped in a jxpath context.
*/
public void doSave(Widget frmModel, JXPathContext jxpc) throws BindingException {
if (this.factory != null) {
jxpc.setFactory(this.factory);
}
Pointer ptr = jxpc.getPointer(this.xpath);
if (ptr.getNode() == null) {
jxpc.createPath(this.xpath);
// Need to recreate the pointer after creating the path
ptr = jxpc.getPointer(this.xpath);
}
JXPathContext subContext = jxpc.getRelativeContext(ptr);
super.doSave(frmModel, subContext);
if (getLogger().isDebugEnabled()) {
getLogger().debug("done saving " + this);
}
}
/** To allow child classes to know which path they bind to */
public String getXPath() {
return this.xpath;
}
public String toString() {
return "ContextJXPathBinding [xpath=" + this.xpath + "]";
}
}