adding pager to manage task resources list

fixing columns in create form wizard to not move around wildy once files are uploaded

fixing some bugs in repeats as a result of the chiba1.3 upgrade.
- xpaths were not being resolved for newly created instances (and shouldn't be)
- xpaths for nested repeats weren't behaving properly

fixed caching issue in edit which was causing the wrong form to be loaded for instance data

using alfresco namespace as defined in NamespaceService

passing namespace uris and prefixes to js to avoid errors if they should change

fixed a bug where the renditions property of forminstancedata was getting duplicate renditions (and thus regenerating way too much stuff when doing an edit)

using an italic 'description not set' when description isn't set in several screens.

using the avm service to set properties in renderingenginetemplateimpl.

upgrading to xalan-2.7.0 in order to be able to use bsf for extension functions.

adding the path that was not found to AVMNotFoundExceptions.  very helpful when debugging.

substantial refactoring of rendering engines in preparation for integration with TemplateService.
 - implementing a common model as a hash of QNames to objects.  for method, providing a simple method wrapper called TemplateProcessorMethod which takes an array of Objects as a parameter, and returns an object.  it is up to the template processors to properly convert arguments.  a QName is used for the variable name rather than a string in order to include a namespace prefix (needed for xsl, and generally better looking).

- for xsl, using javascript bindings for formdatafunctions, which using liveconnect within rhino to call into the xsl rendering engine to evaluate the function.  it ends up generating this js block into a xalan:component within the xsl tempalte:

// gets the handle to the backing java object which can invoke the function based on id
var _xsltp_invoke = java.lang.Class.forName('org.alfresco.web.forms.XSLTRenderingEngine$ProcessorMethodInvoker').newInstance();

// utility to convert js arrays into java
function _xsltp_to_java_array(js_array) {
var java_array = java.lang.reflect.Array.newInstance(java.lang.Object, js_array.length);
for (var i = 0; i < js_array.length; i++) { java_array[i] = js_array[i]; }
return java_array; }

// js handles to each of the form data functions which uses _xsltp_invoke to call the actual method
function _getAVMPath() { return _xsltp_invoke.invokeMethod('_getAVMPath8829055', _xsltp_to_java_array(arguments)); }
function parseXMLDocuments() { return _xsltp_invoke.invokeMethod('parseXMLDocuments12235190', _xsltp_to_java_array(arguments)); }
function parseXMLDocument() { return _xsltp_invoke.invokeMethod('parseXMLDocument15280968', _xsltp_to_java_array(arguments)); }

xml model data is inferred as a root namespace document within the model hash provided.

- for freemarker, things pretty much work as they did before, just i now need to convert values by hand to TemplateModels

fixed a bug with hidden iframe upload.  seems like the complexity of actually cloning the file input element is unnecessary and that simply attaching the node in two places within the dom works just fine.


git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@4764 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
Ariel Backenroth
2007-01-09 14:42:34 +00:00
parent bd9f33e793
commit cabab31ec0
26 changed files with 1083 additions and 728 deletions

View File

@@ -1,77 +0,0 @@
/*
* Copyright (C) 2005 Alfresco, Inc.
*
* Licensed under the Mozilla Public License version 1.1
* with a permitted attribution clause. You may obtain a
* copy of the License at
*
* http://www.alfresco.org/legal/license.txt
*
* 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.alfresco.web.forms;
import java.util.HashMap;
import java.util.Map;
import javax.faces.context.FacesContext;
import org.alfresco.model.WCMModel;
import org.alfresco.repo.avm.AVMNodeConverter;
import org.alfresco.service.cmr.remote.AVMRemote;
import org.alfresco.service.cmr.repository.ContentService;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.web.bean.wcm.AVMConstants;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.jsf.FacesContextUtils;
/**
* Provides helper functions for form data renderers.
*/
public abstract class AbstractRenderingEngine
implements RenderingEngine
{
private static final Log LOGGER = LogFactory.getLog(AbstractRenderingEngine.class);
protected static final String ALFRESCO_NS = "http://www.alfresco.org/alfresco";
protected static final String ALFRESCO_NS_PREFIX = "alfresco";
protected AbstractRenderingEngine()
{
}
protected static AVMRemote getAVMRemote()
{
final FacesContext fc =
FacesContext.getCurrentInstance();
final WebApplicationContext wac =
FacesContextUtils.getRequiredWebApplicationContext(fc);
return (AVMRemote)wac.getBean("avmRemote");
}
protected static FormDataFunctions getFormDataFunctions()
{
return new FormDataFunctions(AbstractRenderingEngine.getAVMRemote());
}
protected HashMap<String, String> getStandardParameters(final FormInstanceData formInstanceData,
final Rendition rendition)
{
final String formInstanceDataAvmPath = formInstanceData.getPath();
final String renditionAvmPath = rendition.getPath();
final HashMap<String, String> parameters = new HashMap<String, String>();
parameters.put("avm_sandbox_url", AVMConstants.buildStoreUrl(formInstanceDataAvmPath));
parameters.put("form_instance_data_file_name", AVMNodeConverter.SplitBase(formInstanceDataAvmPath)[1]);
parameters.put("rendition_file_name", AVMNodeConverter.SplitBase(renditionAvmPath)[1]);
parameters.put("parent_path", AVMNodeConverter.SplitBase(formInstanceDataAvmPath)[0]);
final FacesContext fc = FacesContext.getCurrentInstance();
parameters.put("request_context_path", fc.getExternalContext().getRequestContextPath());
return parameters;
}
}

View File

@@ -114,8 +114,8 @@ public class FormInstanceDataImpl
final PropertyValue pv =
avmService.getNodeProperty(-1, this.getPath(), WCMAppModel.PROP_RENDITIONS);
final Collection<Serializable> renditionPaths = (pv == null
? Collections.EMPTY_LIST
: pv.getCollection(DataTypeDefinition.TEXT));
? Collections.EMPTY_LIST
: pv.getCollection(DataTypeDefinition.TEXT));
final String storeName = AVMConstants.getStoreName(this.getPath());
final List<Rendition> result = new ArrayList<Rendition>(renditionPaths.size());
for (Serializable path : renditionPaths)

View File

@@ -19,16 +19,15 @@ package org.alfresco.web.forms;
import freemarker.ext.dom.NodeModel;
import freemarker.template.*;
import java.io.*;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.*;
import javax.xml.parsers.DocumentBuilder;
import org.alfresco.model.ContentModel;
import org.alfresco.service.cmr.repository.ContentReader;
import org.alfresco.service.cmr.repository.ContentService;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.service.namespace.NamespaceService;
import org.alfresco.service.namespace.QName;
import org.alfresco.web.bean.wcm.AVMConstants;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
@@ -44,25 +43,16 @@ import org.xml.sax.SAXException;
* @author Ariel Backenroth
*/
public class FreeMarkerRenderingEngine
extends AbstractRenderingEngine
implements RenderingEngine
{
private static final Log LOGGER = LogFactory.getLog(FreeMarkerRenderingEngine.class);
public FreeMarkerRenderingEngine()
{
super();
}
public FreeMarkerRenderingEngine() { }
public String getName()
{
return "FreeMarker";
}
public String getName() { return "FreeMarker"; }
public String getDefaultTemplateFileExtension()
{
return "ftl";
}
public String getDefaultTemplateFileExtension() { return "ftl"; }
/**
* Renders the rendition using the configured freemarker template. This
@@ -70,15 +60,13 @@ public class FreeMarkerRenderingEngine
* a variable named alfresco at the root. the alfresco variable contains a hash of
* all parameters and all extension functions.
*/
public void render(final FormInstanceData formInstanceData,
public void render(final Map<QName, Object> model,
final RenderingEngineTemplate ret,
final Rendition rendition)
final OutputStream out)
throws IOException,
RenderingEngine.RenderingException,
SAXException
{
final Map<String, String> parameters =
this.getStandardParameters(formInstanceData, rendition);
final Configuration cfg = new Configuration();
cfg.setTemplateExceptionHandler(TemplateExceptionHandler.RETHROW_HANDLER);
@@ -86,139 +74,12 @@ public class FreeMarkerRenderingEngine
final Template t = new Template("freemarker_template",
new InputStreamReader(ret.getInputStream()),
cfg);
final TemplateHashModel rootModel = this.convertModel(model);
// wrap the xml instance in a model
final TemplateHashModel instanceDataModel = NodeModel.wrap(formInstanceData.getDocument());
// build models for each of the extension functions
final HashMap<String, TemplateMethodModel> methodModels =
new HashMap<String, TemplateMethodModel>(3, 1.0f);
methodModels.put("parseXMLDocument", new TemplateMethodModel()
{
public Object exec(final List args)
throws TemplateModelException
{
try
{
final FormDataFunctions ef = FreeMarkerRenderingEngine.getFormDataFunctions();
final String path =
AVMConstants.buildPath(parameters.get("parent_path"),
(String)args.get(0),
AVMConstants.PathRelation.WEBAPP_RELATIVE);
final Document d = ef.parseXMLDocument(path);
return d != null ? d.getDocumentElement() : null;
}
catch (Exception e)
{
throw new TemplateModelException(e);
}
}
});
methodModels.put("parseXMLDocuments", new TemplateMethodModel()
{
public Object exec(final List args)
throws TemplateModelException
{
try
{
final FormDataFunctions ef = FreeMarkerRenderingEngine.getFormDataFunctions();
final String path =
AVMConstants.buildPath(parameters.get("parent_path"),
args.size() == 1 ? "" : (String)args.get(1),
AVMConstants.PathRelation.WEBAPP_RELATIVE);
final Map<String, Document> resultMap = ef.parseXMLDocuments((String)args.get(0), path);
LOGGER.debug("received " + resultMap.size() + " documents in " + path);
// create a root document for rooting all the results. we do this
// so that each document root element has a common parent node
// and so that xpath axes work properly
final Document rootNodeDocument = XMLUtil.newDocument();
final Element rootNodeDocumentEl =
rootNodeDocument.createElementNS(ALFRESCO_NS,
ALFRESCO_NS_PREFIX + ":file_list");
rootNodeDocumentEl.setAttribute("xmlns:" + ALFRESCO_NS_PREFIX, ALFRESCO_NS);
rootNodeDocument.appendChild(rootNodeDocumentEl);
final List<NodeModel> result = new ArrayList<NodeModel>(resultMap.size());
for (Map.Entry<String, Document> e : resultMap.entrySet())
{
final Element documentEl = e.getValue().getDocumentElement();
documentEl.setAttribute("xmlns:" + ALFRESCO_NS_PREFIX, ALFRESCO_NS);
documentEl.setAttributeNS(ALFRESCO_NS,
ALFRESCO_NS_PREFIX + ":file_name",
e.getKey());
final Node n = rootNodeDocument.importNode(documentEl, true);
rootNodeDocumentEl.appendChild(n);
result.add(NodeModel.wrap(n));
}
return result;
}
catch (Exception e)
{
throw new TemplateModelException(e);
}
}
});
// for debugging
methodModels.put("_getAVMPath", new TemplateMethodModel()
{
public Object exec(final List args)
throws TemplateModelException
{
try
{
return AVMConstants.buildPath(parameters.get("parent_path"),
(String)args.get(0),
AVMConstants.PathRelation.WEBAPP_RELATIVE);
}
catch (Exception e)
{
throw new TemplateModelException(e);
}
}
});
// build a wrapper for the parameters. this also wraps the extension functions
// so they appear in the namespace alfresco.
final TemplateHashModel parameterModel = new SimpleHash(parameters)
{
public TemplateModel get(final String key)
throws TemplateModelException
{
return (methodModels.containsKey(key)
? methodModels.get(key)
: super.get(key));
}
};
// build the root model. anything not in the falsey alfresco namespace will be
// retrieved from the xml file in order to make it behave as close as possible to
// the xsl environment
final TemplateHashModel rootModel = new TemplateHashModel()
{
public TemplateModel get(final String key)
throws TemplateModelException
{
return (ALFRESCO_NS_PREFIX.equals(key)
? parameterModel
: instanceDataModel.get(key));
}
public boolean isEmpty()
{
return false;
}
};
// process the form
final Writer writer = new OutputStreamWriter(rendition.getOutputStream());
try
{
t.process(rootModel, writer);
t.process(rootModel, new OutputStreamWriter(out));
}
catch (final TemplateException te)
{
@@ -227,8 +88,158 @@ public class FreeMarkerRenderingEngine
}
finally
{
writer.flush();
writer.close();
out.flush();
out.close();
}
}
private TemplateHashModel convertModel(final Map<QName, Object> model)
{
final List<TemplateHashModel> rootModelObjects = new LinkedList<TemplateHashModel>();
final SimpleHash result = new SimpleHash()
{
public TemplateModel get(final String key)
throws TemplateModelException
{
TemplateModel result = super.get(key);
if (result == null)
{
for (TemplateHashModel m : rootModelObjects)
{
result = m.get(key);
if (result != null)
{
break;
}
}
}
return result;
}
};
for (final Map.Entry<QName, Object> entry : model.entrySet())
{
final QName qn = entry.getKey();
if (qn.equals(RenderingEngine.ROOT_NAMESPACE))
{
final TemplateModel m = this.convertValue(entry.getValue());
if (m instanceof TemplateHashModel)
{
rootModelObjects.add((TemplateHashModel)m);
}
else
{
throw new IllegalArgumentException("root namespace values must be convertable to " + TemplateHashModel.class.getName() +
". converted " + entry.getValue().getClass().getName() +
" to " + m.getClass().getName() + ".");
}
}
else
{
final String[] splitQName = QName.splitPrefixedQName(qn.toPrefixString());
final String variableName = splitQName[1];
//insert
if (NamespaceService.DEFAULT_PREFIX.equals(splitQName[0]))
{
result.put(variableName, this.convertValue(entry.getValue()));
}
else
{
SimpleHash prefixModel = null;
final String prefix = splitQName[0];
try
{
try
{
prefixModel = (SimpleHash)result.get(prefix);
}
catch (ClassCastException cce)
{
throw new ClassCastException("expected value of " + prefix +
" to be a " + SimpleHash.class.getName() +
". found a " + result.get(prefix).getClass().getName());
}
}
catch (TemplateModelException tme)
{
}
if (prefixModel == null)
{
prefixModel = new SimpleHash();
result.put(prefix, prefixModel);
}
prefixModel.put(variableName,
this.convertValue(entry.getValue()));
}
}
}
return result;
}
private TemplateModel convertValue(final Object value)
{
LOGGER.debug("converting value " + value);
if (value == null)
{
return null;
}
else if (value.getClass().isArray())
{
final Object[] array = (Object[])value;
LOGGER.debug("converting array of " + array.getClass().getComponentType() +
" size " + array.length);
final SimpleSequence result = new SimpleSequence();
for (int i = 0; i < array.length; i++)
{
result.add(this.convertValue(array[i]));
}
return result;
}
else if (value instanceof String)
{
return new SimpleScalar((String)value);
}
else if (value instanceof Number)
{
return new SimpleNumber((Number)value);
}
else if (value instanceof Boolean)
{
return (Boolean)value ? TemplateBooleanModel.TRUE : TemplateBooleanModel.FALSE;
}
else if (value instanceof Document)
{
return NodeModel.wrap((Document)value);
}
else if (value instanceof Node)
{
return NodeModel.wrap((Node)value);
}
else if (value instanceof RenderingEngine.TemplateProcessorMethod)
{
return new TemplateMethodModel()
{
public Object exec(final List args)
throws TemplateModelException
{
try
{
LOGGER.debug("invoking template processor method " + value +
" with " + args.size() + " arguments");
final Object result = ((TemplateProcessorMethod)value).exec(args.toArray(new Object[args.size()]));
return FreeMarkerRenderingEngine.this.convertValue(result);
}
catch (Exception e)
{
throw new TemplateModelException(e);
}
}
};
}
else
{
throw new IllegalArgumentException("unable to convert value " + value.getClass().getName());
}
}
}

View File

@@ -17,6 +17,7 @@
package org.alfresco.web.forms;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.namespace.QName;
import java.io.IOException;
import java.io.OutputStream;
import java.io.Serializable;
@@ -56,6 +57,18 @@ public interface RenderingEngine
/////////////////////////////////////////////////////////////////////////////
public interface TemplateProcessorMethod
extends Serializable
{
public Object exec(final Object[] arguments)
throws Exception;
}
/////////////////////////////////////////////////////////////////////////////
public final static QName ROOT_NAMESPACE =
QName.createQName(null, "root_namespace");
/**
* Returns the rendering engines name.
*
@@ -81,8 +94,8 @@ public interface RenderingEngine
* @param form the form that collected the xml content.
* @param rendition the rendition to serialize to.
*/
public void render(final FormInstanceData formInstanceData,
public void render(final Map<QName, Object> model,
final RenderingEngineTemplate ret,
final Rendition rendition)
final OutputStream out)
throws IOException, RenderingException, SAXException;
}

View File

@@ -78,9 +78,26 @@ public interface RenderingEngineTemplate
/**
* Produces a rendition of the provided formInstanceData.
*
* @param formInstanceData the form instance data for which to produce
* the rendition.
*/
public Rendition render(final FormInstanceData formInstanceData)
throws IOException,
SAXException,
RenderingEngine.RenderingException;
/**
* Produces a rendition of the provided formInstanceData to an existing
* rendition.
*
* @param formInstanceData the form instance data for which to produce
* the rendition.
* @param rendition the rendition to rerender
*/
public void render(final FormInstanceData formInstanceData,
final Rendition rendition)
throws IOException,
SAXException,
RenderingEngine.RenderingException;
}

View File

@@ -30,15 +30,19 @@ import org.alfresco.service.ServiceRegistry;
import org.alfresco.service.cmr.avm.AVMService;
import org.alfresco.service.cmr.dictionary.DataTypeDefinition;
import org.alfresco.service.cmr.repository.*;
import org.alfresco.service.namespace.QName;
import org.alfresco.service.namespace.*;
import org.alfresco.service.cmr.remote.AVMRemote;
import org.alfresco.web.app.Application;
import org.alfresco.web.bean.repository.Repository;
import org.alfresco.web.bean.wcm.AVMConstants;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.w3c.dom.Document;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.jsf.FacesContextUtils;
import org.w3c.dom.*;
import org.xml.sax.SAXException;
/**
* Implementation of a rendering engine template
*/
@@ -101,11 +105,6 @@ public class RenderingEngineTemplateImpl
return this.renditionPropertiesNodeRef;
}
/**
* Provides an input stream to the rendering engine template.
*
* @return the input stream to the rendering engine template.
*/
public InputStream getInputStream()
throws IOException
{
@@ -131,7 +130,8 @@ public class RenderingEngineTemplateImpl
}
/**
* Returns the output path to use for renditions.
* Generates an output path for the rendition by compiling the output path pattern
* as a freemarker template.
*
* @return the output path to use for renditions.
*/
@@ -183,11 +183,6 @@ public class RenderingEngineTemplateImpl
return result;
}
/**
* Returns the mime type to use for generated assets.
*
* @return the mime type to use for generated assets.
*/
public String getMimetypeForRendition()
{
final NodeService nodeService = this.getServiceRegistry().getNodeService();
@@ -195,9 +190,6 @@ public class RenderingEngineTemplateImpl
WCMAppModel.PROP_MIMETYPE_FOR_RENDITION);
}
/**
* Produces a rendition of the provided formInstanceData.
*/
public Rendition render(final FormInstanceData formInstanceData)
throws IOException,
SAXException,
@@ -213,14 +205,44 @@ public class RenderingEngineTemplateImpl
AVMNodeConverter.SplitBase(renditionAvmPath)[1]);
if (LOGGER.isDebugEnabled())
LOGGER.debug("Created file node for file: " + renditionAvmPath);
avmService.addAspect(renditionAvmPath, WCMAppModel.ASPECT_FORM_INSTANCE_DATA);
avmService.addAspect(renditionAvmPath, ContentModel.ASPECT_TITLED);
avmService.addAspect(renditionAvmPath, WCMAppModel.ASPECT_RENDITION);
final PropertyValue pv =
avmService.getNodeProperty(-1, formInstanceData.getPath(), WCMAppModel.PROP_RENDITIONS);
final Collection<Serializable> renditions = (pv == null
? new HashSet<Serializable>()
: pv.getCollection(DataTypeDefinition.TEXT));
renditions.add(AVMConstants.getStoreRelativePath(renditionAvmPath));
avmService.setNodeProperty(formInstanceData.getPath(),
WCMAppModel.PROP_RENDITIONS,
new PropertyValue(DataTypeDefinition.TEXT,
(Serializable)renditions));
}
final Rendition result = new RenditionImpl(AVMNodeConverter.ToNodeRef(-1, renditionAvmPath));
this.getRenderingEngine().render(formInstanceData, this, result);
this.render(formInstanceData, result);
return result;
}
avmService.addAspect(renditionAvmPath, WCMAppModel.ASPECT_FORM_INSTANCE_DATA);
avmService.addAspect(renditionAvmPath, ContentModel.ASPECT_TITLED);
avmService.addAspect(renditionAvmPath, WCMAppModel.ASPECT_RENDITION);
public void render(final FormInstanceData formInstanceData,
final Rendition rendition)
throws IOException,
SAXException,
RenderingEngine.RenderingException
{
final OutputStream out = rendition.getOutputStream();
try
{
this.getRenderingEngine().render(this.buildModel(formInstanceData, rendition),
this,
out);
}
finally
{
out.close();
}
final Map<QName, PropertyValue> props = new HashMap<QName, PropertyValue>(5, 1.0f);
props.put(WCMAppModel.PROP_PARENT_FORM_NAME,
@@ -228,13 +250,13 @@ public class RenderingEngineTemplateImpl
formInstanceData.getForm().getName()));
props.put(ContentModel.PROP_TITLE,
new PropertyValue(DataTypeDefinition.TEXT,
AVMNodeConverter.SplitBase(renditionAvmPath)[1]));
AVMNodeConverter.SplitBase(rendition.getPath())[1]));
final ResourceBundle bundle = Application.getBundle(FacesContext.getCurrentInstance());
props.put(ContentModel.PROP_DESCRIPTION,
new PropertyValue(DataTypeDefinition.TEXT,
MessageFormat.format(bundle.getString("default_rendition_description"),
this.getTitle(),
AVMConstants.getSandboxRelativePath(renditionAvmPath))));
AVMConstants.getSandboxRelativePath(rendition.getPath()))));
props.put(WCMAppModel.PROP_PARENT_RENDERING_ENGINE_TEMPLATE,
new PropertyValue(DataTypeDefinition.NODE_REF,
this.nodeRef));
@@ -246,19 +268,187 @@ public class RenderingEngineTemplateImpl
new PropertyValue(DataTypeDefinition.TEXT,
AVMConstants.getStoreRelativePath(formInstanceData.getPath())));
avmService.setNodeProperties(renditionAvmPath, props);
final PropertyValue pv =
avmService.getNodeProperty(-1, formInstanceData.getPath(), WCMAppModel.PROP_RENDITIONS);
Collection<Serializable> renditions = (pv == null
? new LinkedList<Serializable>()
: pv.getCollection(DataTypeDefinition.TEXT));
renditions.add(AVMConstants.getStoreRelativePath(renditionAvmPath));
avmService.setNodeProperty(formInstanceData.getPath(),
WCMAppModel.PROP_RENDITIONS,
new PropertyValue(DataTypeDefinition.TEXT,
(Serializable)renditions));
return result;
final AVMService avmService = this.getServiceRegistry().getAVMService();
avmService.setNodeProperties(rendition.getPath(), props);
}
/**
* Builds the model to pass to the rendering engine.
*/
protected Map<QName, Object> buildModel(final FormInstanceData formInstanceData,
final Rendition rendition)
throws IOException,
SAXException
{
final DynamicNamespacePrefixResolver namespacePrefixResolver =
new DynamicNamespacePrefixResolver();
namespacePrefixResolver.registerNamespace(NamespaceService.ALFRESCO_PREFIX,
NamespaceService.ALFRESCO_URI);
final String formInstanceDataAvmPath = formInstanceData.getPath();
final String renditionAvmPath = rendition.getPath();
final String parentPath = AVMNodeConverter.SplitBase(formInstanceDataAvmPath)[0];
final String sandboxUrl = AVMConstants.buildStoreUrl(formInstanceDataAvmPath);
final HashMap<QName, Object> model = new HashMap<QName, Object>();
// add simple scalar parameters
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"avm_sandbox_url",
namespacePrefixResolver),
sandboxUrl);
model.put(XSLTRenderingEngine.PROP_URI_RESOLVER_BASE_URI,
sandboxUrl);
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"form_instance_data_file_name",
namespacePrefixResolver),
AVMNodeConverter.SplitBase(formInstanceDataAvmPath)[1]);
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"rendition_file_name",
namespacePrefixResolver),
AVMNodeConverter.SplitBase(renditionAvmPath)[1]);
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"parent_path",
namespacePrefixResolver),
parentPath);
final FacesContext fc = FacesContext.getCurrentInstance();
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"request_context_path",
namespacePrefixResolver),
fc.getExternalContext().getRequestContextPath());
// add methods
final FormDataFunctions fdf = this.getFormDataFunctions();
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"parseXMLDocument",
namespacePrefixResolver),
new RenderingEngine.TemplateProcessorMethod()
{
public Object exec(final Object[] arguments)
throws IOException,
SAXException
{
if (arguments.length != 1)
{
throw new IllegalArgumentException("expected 1 argument to parseXMLDocument. got " +
arguments.length);
}
if (! (arguments[0] instanceof String))
{
throw new ClassCastException("expected arguments[0] to be a " + String.class.getName() +
". got a " + arguments[0].getClass().getName() + ".");
}
String path = (String)arguments[0];
path = AVMConstants.buildPath(parentPath,
path,
AVMConstants.PathRelation.WEBAPP_RELATIVE);
LOGGER.debug("tpm_parseXMLDocument('" + path +
"'), parentPath = " + parentPath);
final Document d = fdf.parseXMLDocument(path);
return d != null ? d.getDocumentElement() : null;
}
});
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"parseXMLDocuments",
namespacePrefixResolver),
new RenderingEngine.TemplateProcessorMethod()
{
public Object exec(final Object[] arguments)
throws IOException,
SAXException
{
if (arguments.length > 1)
{
throw new IllegalArgumentException("expected zero or one arguments to parseXMLDocuments. got " +
arguments.length);
}
if (! (arguments[0] instanceof String))
{
throw new ClassCastException("expected arguments[0] to be a " + String.class.getName() +
". got a " + arguments[0].getClass().getName() + ".");
}
if (arguments.length == 2 && ! (arguments[1] instanceof String))
{
throw new ClassCastException("expected arguments[1] to be a " + String.class.getName() +
". got a " + arguments[1].getClass().getName() + ".");
}
String path = arguments.length == 2 ? (String)arguments[1] : "";
path = AVMConstants.buildPath(parentPath,
path,
AVMConstants.PathRelation.WEBAPP_RELATIVE);
final String formName = (String)arguments[0];
LOGGER.debug("tpm_parseXMLDocuments('" + formName + "','" + path +
"'), parentPath = " + parentPath);
final Map<String, Document> resultMap = fdf.parseXMLDocuments(formName, path);
LOGGER.debug("received " + resultMap.size() +
" documents in " + path +
" with form name " + formName);
// create a root document for rooting all the results. we do this
// so that each document root element has a common parent node
// and so that xpath axes work properly
final Document rootNodeDocument = XMLUtil.newDocument();
final Element rootNodeDocumentEl =
rootNodeDocument.createElementNS(NamespaceService.ALFRESCO_URI,
NamespaceService.ALFRESCO_PREFIX + ":file_list");
rootNodeDocumentEl.setAttribute("xmlns:" + NamespaceService.ALFRESCO_PREFIX,
NamespaceService.ALFRESCO_URI);
rootNodeDocument.appendChild(rootNodeDocumentEl);
final List<Node> result = new ArrayList<Node>(resultMap.size());
for (Map.Entry<String, Document> e : resultMap.entrySet())
{
final Element documentEl = e.getValue().getDocumentElement();
documentEl.setAttribute("xmlns:" + NamespaceService.ALFRESCO_PREFIX,
NamespaceService.ALFRESCO_URI);
documentEl.setAttributeNS(NamespaceService.ALFRESCO_URI,
NamespaceService.ALFRESCO_PREFIX + ":file_name",
e.getKey());
final Node n = rootNodeDocument.importNode(documentEl, true);
rootNodeDocumentEl.appendChild(n);
result.add(n);
}
return result.toArray(new Node[result.size()]);
}
});
model.put(QName.createQName(NamespaceService.ALFRESCO_PREFIX,
"_getAVMPath",
namespacePrefixResolver),
new RenderingEngine.TemplateProcessorMethod()
{
public Object exec(final Object[] arguments)
{
if (arguments.length != 1)
{
throw new IllegalArgumentException("expected one argument to _getAVMPath. got " +
arguments.length);
}
if (! (arguments[0] instanceof String))
{
throw new ClassCastException("expected arguments[0] to be a " + String.class.getName() +
". got a " + arguments[0].getClass().getName() + ".");
}
final String path = (String)arguments[0];
LOGGER.debug("tpm_getAVMPAth('" + path + "'), parentPath = " + parentPath);
return AVMConstants.buildPath(parentPath,
path,
AVMConstants.PathRelation.WEBAPP_RELATIVE);
}
});
// add the xml document
model.put(RenderingEngine.ROOT_NAMESPACE, formInstanceData.getDocument());
return model;
}
protected static FormDataFunctions getFormDataFunctions()
{
final FacesContext fc = FacesContext.getCurrentInstance();
final WebApplicationContext wac =
FacesContextUtils.getRequiredWebApplicationContext(fc);
return new FormDataFunctions((AVMRemote)wac.getBean("avmRemote"));
}
private ServiceRegistry getServiceRegistry()

View File

@@ -161,14 +161,13 @@ public class RenditionImpl
this.regenerate(this.getPrimaryFormInstanceData());
}
@Deprecated
public void regenerate(final FormInstanceData formInstanceData)
throws IOException,
RenderingEngine.RenderingException,
SAXException
{
final RenderingEngineTemplate ret = this.getRenderingEngineTemplate();
final RenderingEngine engine = ret.getRenderingEngine();
engine.render(formInstanceData, ret, this);
this.getRenderingEngineTemplate().render(formInstanceData, this);
}
private ServiceRegistry getServiceRegistry()

View File

@@ -43,6 +43,7 @@ import org.xml.sax.SAXException;
*/
public class XMLUtil
{
private static final Log LOGGER = LogFactory.getLog(XMLUtil.class);
private static DocumentBuilder documentBuilder;

View File

@@ -22,6 +22,7 @@ import javax.xml.transform.Result;
import javax.xml.transform.Source;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.sax.SAXResult;
import org.alfresco.service.namespace.QName;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.fop.apps.FOPException;
@@ -86,9 +87,10 @@ public class XSLFORenderingEngine
return "fo";
}
public void render(final FormInstanceData formInstanceData,
@Override
public void render(final Map<QName, Object> model,
final RenderingEngineTemplate ret,
final Rendition rendition)
final OutputStream out)
throws IOException,
RenderingEngine.RenderingException,
SAXException
@@ -100,16 +102,14 @@ public class XSLFORenderingEngine
throw new RenderingEngine.RenderingException("mimetype " + ret.getMimetypeForRendition() +
" is not supported by " + this.getName());
}
final OutputStream out = rendition.getOutputStream();
try
{
final FopFactory fopFactory = FopFactory.newInstance();
final FOUserAgent foUserAgent = fopFactory.newFOUserAgent();
final Fop fop = fopFactory.newFop(mimetype, foUserAgent, out);
// Resulting SAX events (the generated FO) must be piped through to FOP
super.render(new DOMSource(formInstanceData.getDocument()),
super.render(model,
ret,
this.getStandardParameters(formInstanceData, rendition),
new SAXResult(fop.getDefaultHandler()));
}

View File

@@ -35,13 +35,17 @@ import javax.xml.transform.stream.StreamSource;
import org.alfresco.service.cmr.repository.ContentService;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.service.namespace.NamespaceService;
import org.alfresco.service.namespace.QName;
import org.alfresco.web.bean.wcm.AVMConstants;
import org.alfresco.web.forms.XMLUtil;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.xalan.extensions.ExpressionContext;
import org.apache.xpath.objects.XObject;
import org.apache.xml.utils.QName;
import org.apache.xml.dtm.ref.DTMNodeProxy;
import org.apache.xml.utils.Constants;
//import org.apache.xml.utils.QName;
import org.w3c.dom.*;
import org.w3c.dom.traversal.NodeFilter;
import org.w3c.dom.traversal.NodeIterator;
@@ -55,187 +59,176 @@ import org.apache.bsf.BSFManager;
* @author Ariel Backenroth
*/
public class XSLTRenderingEngine
extends AbstractRenderingEngine
implements RenderingEngine
{
private static final Log LOGGER = LogFactory.getLog(XSLTRenderingEngine.class);
/////////////////////////////////////////////////////////////////////////////
public XSLTRenderingEngine()
public static class ProcessorMethodInvoker
{
super();
}
private final static HashMap<String, TemplateProcessorMethod> PROCESSOR_METHODS =
new HashMap<String, TemplateProcessorMethod>();
public ProcessorMethodInvoker() { }
public String getName()
{
return "XSLT";
}
public String getDefaultTemplateFileExtension()
{
return "xsl";
}
protected static String toAVMPath(final ExpressionContext ec, String path)
throws TransformerException
{
final XObject o = ec.getVariableOrParam(new QName(ALFRESCO_NS, ALFRESCO_NS_PREFIX, "parent_path"));
return o == null ? null : AVMConstants.buildPath(o.toString(),
path,
AVMConstants.PathRelation.WEBAPP_RELATIVE);
}
/**
* Adapter function used by the xsl tempalte to retrieve an xml asset at the given
* path.
*
* @return the document element for the xml asset at the given path.
*/
public static Node parseXMLDocument(final ExpressionContext ec, final String path)
throws TransformerException,
IOException,
SAXException
{
final FormDataFunctions ef = XSLTRenderingEngine.getFormDataFunctions();
final Document d = ef.parseXMLDocument(XSLTRenderingEngine.toAVMPath(ec, path));
return d != null ? d.getDocumentElement() : null;
}
/**
* Adapter function used by the xsl tempalte to retrieve a xml assets in the
* current directory.
*/
public static NodeIterator parseXMLDocuments(final ExpressionContext ec,
final String formName)
throws TransformerException,
IOException,
SAXException
{
return XSLTRenderingEngine.parseXMLDocuments(ec, formName, "");
}
/**
* Adapter function used by the xsl tempalte to retrieve a xml assets at
* the given path.
*
* @return an iterator of the document elements for each of the xml
* assets at the given path. In order to enable xpath expressions to
* properly access siblings, each root element is rooted at a node named
* file-list in the alfresco namespace.
*/
public static NodeIterator parseXMLDocuments(final ExpressionContext ec,
final String formName,
String path)
throws TransformerException,
IOException,
SAXException
{
final FormDataFunctions ef = XSLTRenderingEngine.getFormDataFunctions();
path = XSLTRenderingEngine.toAVMPath(ec, path);
final Map<String, Document> resultMap = ef.parseXMLDocuments(formName, path);
if (LOGGER.isDebugEnabled())
LOGGER.debug("received " + resultMap.size() + " documents in " + path);
// create a root document for rooting all the results. we do this
// so that each document root element has a common parent node
// and so that xpath axes work properly
final Document rootNodeDocument = XMLUtil.newDocument();
final Element rootNodeDocumentEl =
rootNodeDocument.createElementNS(ALFRESCO_NS,
ALFRESCO_NS_PREFIX + ":file_list");
rootNodeDocumentEl.setAttribute("xmlns:" + ALFRESCO_NS_PREFIX, ALFRESCO_NS);
rootNodeDocument.appendChild(rootNodeDocumentEl);
final List<Node> documents = new ArrayList<Node>(resultMap.size());
for (Map.Entry<String, Document> mapEntry : resultMap.entrySet())
private Object[] convertArguments(final Object[] arguments)
{
final Element documentEl = mapEntry.getValue().getDocumentElement();
documentEl.setAttributeNS(ALFRESCO_NS,
ALFRESCO_NS_PREFIX + ":file_name",
mapEntry.getKey());
final Node n = rootNodeDocument.importNode(documentEl, true);
documents.add(n);
rootNodeDocumentEl.appendChild(n);
final List result = new LinkedList();
for (int i = 0; i < arguments.length; i++)
{
LOGGER.debug("args[" + i + "] = " + arguments[i] +
"(" + (arguments[i] != null
? arguments[i].getClass().getName()
: "null") + ")");
if (arguments[i] == null ||
arguments[i] instanceof String ||
arguments[i] instanceof Number)
{
result.add(arguments[i]);
}
else if (arguments[i] instanceof DTMNodeProxy)
{
result.add(((DTMNodeProxy)arguments[i]).getStringValue());
}
else if (arguments[i] instanceof Node)
{
LOGGER.debug("node type is " + ((Node)arguments[i]).getNodeType() +
" content " + ((Node)arguments[i]).getTextContent());
result.add(((Node)arguments[i]).getNodeValue());
}
else if (arguments[i] instanceof NodeIterator)
{
Node n = ((NodeIterator)arguments[i]).nextNode();
while (n != null)
{
LOGGER.debug("iterated to node " + n + " type " + n.getNodeType() +
" value " + n.getNodeValue() +
" tc " + n.getTextContent() +
" nn " + n.getNodeName() +
" sv " + ((org.apache.xml.dtm.ref.DTMNodeProxy)n).getStringValue());
if (n instanceof DTMNodeProxy)
{
result.add(((DTMNodeProxy)n).getStringValue());
}
else
{
result.add(n);
}
n = ((NodeIterator)arguments[i]).nextNode();
}
}
else
{
throw new IllegalArgumentException("unable to convert argument " + arguments[i]);
}
}
return result.toArray(new Object[result.size()]);
}
return new NodeIterator()
public Object invokeMethod(final String id, Object[] arguments)
throws Exception
{
private int index = 0;
private boolean detached = false;
public void detach()
{
if (LOGGER.isDebugEnabled())
LOGGER.debug("detaching NodeIterator");
resultMap.clear();
documents.clear();
this.detached = true;
}
public boolean getExpandEntityReferences()
{
return true;
if (!PROCESSOR_METHODS.containsKey(id))
{
throw new NullPointerException("unable to find method " + id);
}
public NodeFilter getFilter()
{
return new NodeFilter()
final TemplateProcessorMethod method = PROCESSOR_METHODS.get(id);
arguments = this.convertArguments(arguments);
LOGGER.debug("invoking " + id + " with " + arguments.length);
Object result = method.exec(arguments);
LOGGER.debug(id + " returned a " + result);
if (result == null)
{
return null;
}
else if (result.getClass().isArray() &&
Node.class.isAssignableFrom(result.getClass().getComponentType()))
{
LOGGER.debug("converting " + result + " to a node iterator");
final Node[] array = (Node[])result;
return new NodeIterator()
{
public short acceptNode(final Node n)
private int index = 0;
private boolean detached = false;
public void detach()
{
if (LOGGER.isDebugEnabled())
LOGGER.debug("detaching NodeIterator");
this.detached = true;
}
public boolean getExpandEntityReferences() { return true; }
public int getWhatToShow() { return NodeFilter.SHOW_ALL; }
public Node getRoot()
{
return NodeFilter.FILTER_ACCEPT;
return (array.length == 0
? null
: array[0].getOwnerDocument().getDocumentElement());
}
public NodeFilter getFilter()
{
return new NodeFilter()
{
public short acceptNode(final Node n)
{
return NodeFilter.FILTER_ACCEPT;
}
};
}
public Node nextNode()
throws DOMException
{
if (LOGGER.isDebugEnabled())
LOGGER.debug("NodeIterator.nextNode(" + index + ")");
if (this.detached)
throw new DOMException(DOMException.INVALID_STATE_ERR, null);
return index == array.length ? null : array[index++];
}
public Node previousNode()
throws DOMException
{
if (LOGGER.isDebugEnabled())
LOGGER.debug("NodeIterator.previousNode(" + index + ")");
if (this.detached)
throw new DOMException(DOMException.INVALID_STATE_ERR, null);
return index == -1 ? null : array[index--];
}
};
}
public Node getRoot()
else if (result instanceof String ||
result instanceof Number ||
result instanceof Node)
{
return rootNodeDocumentEl;
LOGGER.debug("returning " + result + " as is");
return result;
}
public int getWhatToShow()
else
{
return NodeFilter.SHOW_ALL;
throw new IllegalArgumentException("unable to convert " + result.getClass().getName());
}
public Node nextNode()
throws DOMException
{
if (LOGGER.isDebugEnabled())
LOGGER.debug("NodeIterator.nextNode(" + index + ")");
if (this.detached)
throw new DOMException(DOMException.INVALID_STATE_ERR, null);
if (index == documents.size())
return null;
return documents.get(index++);
}
public Node previousNode()
throws DOMException
{
if (LOGGER.isDebugEnabled())
LOGGER.debug("NodeIterator.previousNode(" + index + ")");
if (this.detached)
throw new DOMException(DOMException.INVALID_STATE_ERR, null);
if (index == -1)
return null;
return documents.get(index--);
}
};
}
}
/**
* for debugging only. provides the absolute avm path for the given
* path.
*/
public static String _getAVMPath(final ExpressionContext ec,
final String path)
throws TransformerException,
IOException,
SAXException
{
final FormDataFunctions ef = XSLTRenderingEngine.getFormDataFunctions();
return XSLTRenderingEngine.toAVMPath(ec, path);
}
/////////////////////////////////////////////////////////////////////////////
private static final Log LOGGER = LogFactory.getLog(XSLTRenderingEngine.class);
public static final QName PROP_URI_RESOLVER_BASE_URI =
QName.createQName(NamespaceService.ALFRESCO_URI, "xslt_resolver_base_uri");
public XSLTRenderingEngine() { }
public String getName() { return "XSLT"; }
public String getDefaultTemplateFileExtension() { return "xsl"; }
/**
* Adds a script element to the xsl which makes static methods on this
@@ -243,22 +236,75 @@ public class XSLTRenderingEngine
*
* @param xslTemplate the xsl template
*/
protected void addScript(final Document xslTemplate)
protected List<String> addScripts(final Map<QName, Object> model,
final Document xslTemplate)
{
final Map<QName, List<Map.Entry<QName, Object>>> methods =
new HashMap<QName, List<Map.Entry<QName, Object>>>();
for (final Map.Entry<QName, Object> entry : model.entrySet())
{
if (entry.getValue() instanceof TemplateProcessorMethod)
{
final String prefix = QName.splitPrefixedQName(entry.getKey().toPrefixString())[0];
final QName qn = QName.createQName(entry.getKey().getNamespaceURI(),
prefix);
if (!methods.containsKey(qn))
{
methods.put(qn, new LinkedList());
}
methods.get(qn).add(entry);
}
}
final Element docEl = xslTemplate.getDocumentElement();
final String XALAN_NS = "http://xml.apache.org/xalan";
final String XALAN_NS = Constants.S_BUILTIN_EXTENSIONS_URL;
final String XALAN_NS_PREFIX = "xalan";
docEl.setAttribute("xmlns:" + XALAN_NS_PREFIX, XALAN_NS);
docEl.setAttribute("xmlns:" + ALFRESCO_NS_PREFIX, ALFRESCO_NS);
final Element compEl = xslTemplate.createElementNS(XALAN_NS, XALAN_NS_PREFIX + ":component");
compEl.setAttribute("prefix", "alfresco");
docEl.appendChild(compEl);
final Element scriptEl = xslTemplate.createElementNS(XALAN_NS, XALAN_NS_PREFIX + ":script");
scriptEl.setAttribute("lang", "javaclass");
scriptEl.setAttribute("src", XALAN_NS_PREFIX + "://" + this.getClass().getName());
compEl.appendChild(scriptEl);
final List<String> result = new LinkedList<String>();
for (QName ns : methods.keySet())
{
final String prefix = ns.getLocalName();
docEl.setAttribute("xmlns:" + prefix, ns.getNamespaceURI());
final Element compEl = xslTemplate.createElementNS(XALAN_NS,
XALAN_NS_PREFIX + ":component");
compEl.setAttribute("prefix", prefix);
docEl.appendChild(compEl);
String functionNames = null;
final Element scriptEl = xslTemplate.createElementNS(XALAN_NS,
XALAN_NS_PREFIX + ":script");
scriptEl.setAttribute("lang", "javascript");
final StringBuilder js =
new StringBuilder("var _xsltp_invoke = java.lang.Class.forName('" + ProcessorMethodInvoker.class.getName() +
"').newInstance();\n" +
"function _xsltp_to_java_array(js_array) {\n" +
"var java_array = java.lang.reflect.Array.newInstance(java.lang.Object, js_array.length);\n" +
"for (var i = 0; i < js_array.length; i++) { java_array[i] = js_array[i]; }\n" +
"return java_array; }\n");
for (final Map.Entry<QName, Object> entry : methods.get(ns))
{
if (functionNames == null)
{
functionNames = entry.getKey().getLocalName();
}
else
{
functionNames += " " + entry.getKey().getLocalName();
}
final String id = entry.getKey().getLocalName() + entry.getValue().hashCode();
js.append("function " + entry.getKey().getLocalName() +
"() { return _xsltp_invoke.invokeMethod('" + id +
"', _xsltp_to_java_array(arguments)); }\n");
ProcessorMethodInvoker.PROCESSOR_METHODS.put(id, (TemplateProcessorMethod)entry.getValue());
result.add(id);
}
LOGGER.debug("generated JavaScript bindings:\n" + js);
scriptEl.appendChild(xslTemplate.createTextNode(js.toString()));
compEl.setAttribute("functions", functionNames);
compEl.appendChild(scriptEl);
}
return result;
}
/**
@@ -268,47 +314,58 @@ public class XSLTRenderingEngine
* @param parameters the variables to place within the xsl template
* @param xslTemplate the xsl template
*/
protected void addParameters(final Map<String, String> parameters,
final Document xslTemplate)
protected void addParameters(final Map<QName, Object> model,
final Document xslTemplate)
{
final Element docEl = xslTemplate.getDocumentElement();
final String XSL_NS = docEl.getNamespaceURI();
final String XSL_NS_PREFIX = docEl.getPrefix();
for (Map.Entry<String, String> e : parameters.entrySet())
for (Map.Entry<QName, Object> e : model.entrySet())
{
if (RenderingEngine.ROOT_NAMESPACE.equals(e.getKey()))
{
continue;
}
final Element el = xslTemplate.createElementNS(XSL_NS, XSL_NS_PREFIX + ":variable");
el.setAttribute("name", ALFRESCO_NS_PREFIX + ':' + e.getKey());
el.appendChild(xslTemplate.createTextNode(e.getValue()));
docEl.insertBefore(el, docEl.getFirstChild());
el.setAttribute("name", e.getKey().toPrefixString());
final Object o = e.getValue();
if (o instanceof String || o instanceof Number || o instanceof Boolean)
{
el.appendChild(xslTemplate.createTextNode(o.toString()));
docEl.insertBefore(el, docEl.getFirstChild());
}
}
}
protected Source getXMLSource(final Map<QName, Object> model)
{
if (!model.containsKey(RenderingEngine.ROOT_NAMESPACE))
{
return null;
}
final Object o = model.get(RenderingEngine.ROOT_NAMESPACE);
if (!(o instanceof Document))
{
throw new IllegalArgumentException("expected root namespace object to be a " + Document.class.getName() +
". found a " + o.getClass().getName());
}
return new DOMSource((Document)o);
}
public void render(final FormInstanceData formInstanceData,
public void render(final Map<QName, Object> model,
final RenderingEngineTemplate ret,
final Rendition rendition)
final OutputStream out)
throws IOException,
RenderingEngine.RenderingException,
SAXException
{
final OutputStream out = rendition.getOutputStream();
try
{
this.render(new DOMSource(formInstanceData.getDocument()),
ret,
this.getStandardParameters(formInstanceData, rendition),
new StreamResult(out));
}
finally
{
out.close();
}
this.render(model, ret, new StreamResult(out));
}
protected void render(final Source formInstanceDataSource,
final RenderingEngineTemplate ret,
final Map<String, String> parameters,
final Result result)
public void render(final Map<QName, Object> model,
final RenderingEngineTemplate ret,
final Result result)
throws IOException,
RenderingEngine.RenderingException,
SAXException
@@ -324,8 +381,10 @@ public class XSLTRenderingEngine
{
throw new RenderingEngine.RenderingException(sax);
}
this.addScript(xslTemplate);
this.addParameters(parameters, xslTemplate);
this.addScripts(model, xslTemplate);
this.addParameters(model, xslTemplate);
Source xmlSource = this.getXMLSource(model);
Transformer t = null;
try
@@ -344,21 +403,25 @@ public class XSLTRenderingEngine
// web application
t.setURIResolver(new URIResolver()
{
public Source resolve(final String href, final String base)
public Source resolve(final String href, String base)
throws TransformerException
{
// XXXarielb - dirty - fix this
final String sandBoxUrl = (String)parameters.get("avm_sandbox_url");
LOGGER.debug("request to resolve href " + href +
" using base " + base);
if (model.containsKey(PROP_URI_RESOLVER_BASE_URI))
{
base = (String)model.get(PROP_URI_RESOLVER_BASE_URI);
LOGGER.debug("overriding base with " + base);
}
URI uri = null;
try
{
uri = new URI(sandBoxUrl + href);
uri = new URI(base + href);
}
catch (URISyntaxException e)
{
throw new TransformerException("unable to create uri " +
sandBoxUrl + href,
throw new TransformerException("unable to create uri " + base + href,
e);
}
try
@@ -379,11 +442,16 @@ public class XSLTRenderingEngine
try
{
t.transform(formInstanceDataSource, result);
t.transform(xmlSource, result);
}
catch (TransformerException e)
catch (TransformerException te)
{
LOGGER.error(e.getMessageAndLocation());
LOGGER.error(te.getMessageAndLocation());
throw new RenderingEngine.RenderingException(te);
}
catch (Exception e)
{
LOGGER.error("unexpected error " + e);
throw new RenderingEngine.RenderingException(e);
}
}

View File

@@ -21,6 +21,7 @@ import java.io.IOException;
import java.io.StringWriter;
import java.util.*;
import javax.xml.transform.*;
import org.alfresco.service.namespace.NamespaceService;
import org.alfresco.web.forms.XMLUtil;
import org.apache.commons.jxpath.JXPathContext;
import org.apache.commons.jxpath.Pointer;
@@ -70,16 +71,7 @@ public class SchemaFormBuilder
/////////////////////////////////////////////////////////////////////////////
public final static Log LOGGER =
LogFactory.getLog(SchemaFormBuilder.class);
/** Alfresco namespace declaration. */
private static final String ALFRESCO_NS =
"http://www.alfresco.org/alfresco";
/** Alfresco prefix */
private static final String ALFRESCO_PREFIX = "alfresco";
private final static Log LOGGER = LogFactory.getLog(SchemaFormBuilder.class);
private static final String PROPERTY_PREFIX =
"http://www.chiba.org/properties/schemaFormBuilder/";
@@ -433,10 +425,12 @@ public class SchemaFormBuilder
{
final JXPathContext prototypeContext =
JXPathContext.newContext(prototypeDocumentElement);
prototypeContext.registerNamespace("alfresco", SchemaFormBuilder.ALFRESCO_NS);
prototypeContext.registerNamespace(NamespaceService.ALFRESCO_PREFIX,
NamespaceService.ALFRESCO_URI);
final JXPathContext instanceContext =
JXPathContext.newContext(instanceDocumentElement);
instanceContext.registerNamespace("alfresco", SchemaFormBuilder.ALFRESCO_NS);
instanceContext.registerNamespace(NamespaceService.ALFRESCO_PREFIX,
NamespaceService.ALFRESCO_URI);
class PrototypeInsertionData
{
@@ -457,7 +451,7 @@ public class SchemaFormBuilder
new LinkedList<PrototypeInsertionData>();
final Iterator it =
prototypeContext.iteratePointers("//*[@" + SchemaFormBuilder.ALFRESCO_PREFIX +
prototypeContext.iteratePointers("//*[@" + NamespaceService.ALFRESCO_PREFIX +
":prototype='true']");
while (it.hasNext())
{
@@ -528,11 +522,11 @@ public class SchemaFormBuilder
{
for (Element e : l)
{
if (e.hasAttributeNS(SchemaFormBuilder.ALFRESCO_NS, "prototype"))
if (e.hasAttributeNS(NamespaceService.ALFRESCO_URI, "prototype"))
{
assert "true".equals(e.getAttributeNS(SchemaFormBuilder.ALFRESCO_NS,
assert "true".equals(e.getAttributeNS(NamespaceService.ALFRESCO_URI,
"prototype"));
e.removeAttributeNS(SchemaFormBuilder.ALFRESCO_NS, "prototype");
e.removeAttributeNS(NamespaceService.ALFRESCO_URI, "prototype");
if (l.getLast().equals(e))
{
@@ -1687,8 +1681,8 @@ public class SchemaFormBuilder
final Element e = (Element)newDefaultInstanceElement.cloneNode(true);
if (i == elementOccurs.minimum)
{
e.setAttributeNS(SchemaFormBuilder.ALFRESCO_NS,
SchemaFormBuilder.ALFRESCO_PREFIX + ":prototype",
e.setAttributeNS(NamespaceService.ALFRESCO_URI,
NamespaceService.ALFRESCO_PREFIX + ":prototype",
"true");
}
defaultInstanceElement.appendChild(e);
@@ -1990,7 +1984,7 @@ public class SchemaFormBuilder
formControl.appendChild(alertElement);
this.setXFormsId(alertElement);
String alert = SchemaFormBuilder.extractPropertyFromAnnotation(ALFRESCO_NS,
String alert = SchemaFormBuilder.extractPropertyFromAnnotation(NamespaceService.ALFRESCO_URI,
"alert",
this.getAnnotation(owner),
resourceBundle);
@@ -2082,8 +2076,8 @@ public class SchemaFormBuilder
NamespaceConstants.XMLSCHEMA_INSTANCE_PREFIX,
NamespaceConstants.XMLSCHEMA_INSTANCE_NS);
this.addNamespace(envelopeElement,
SchemaFormBuilder.ALFRESCO_PREFIX,
SchemaFormBuilder.ALFRESCO_NS);
NamespaceService.ALFRESCO_PREFIX,
NamespaceService.ALFRESCO_URI);
//base
if (this.base != null && this.base.length() != 0)
@@ -2150,7 +2144,7 @@ public class SchemaFormBuilder
final XSAnnotation annotation,
final ResourceBundle resourceBundle)
{
final String s = SchemaFormBuilder.extractPropertyFromAnnotation(ALFRESCO_NS,
final String s = SchemaFormBuilder.extractPropertyFromAnnotation(NamespaceService.ALFRESCO_URI,
"label",
annotation,
resourceBundle);
@@ -2540,7 +2534,7 @@ public class SchemaFormBuilder
final XSAnnotation annotation = this.getAnnotation(node);
if (annotation == null)
return null;
final String s = this.extractPropertyFromAnnotation(ALFRESCO_NS,
final String s = this.extractPropertyFromAnnotation(NamespaceService.ALFRESCO_URI,
"hint",
annotation,
resourceBundle);
@@ -2666,16 +2660,16 @@ public class SchemaFormBuilder
{
//if 0 or 1 -> no constraint (managed by "required")
minConstraint = "count(.) >= " + o.minimum;
bindElement.setAttributeNS(ALFRESCO_NS,
ALFRESCO_PREFIX + ":minimum",
bindElement.setAttributeNS(NamespaceService.ALFRESCO_URI,
NamespaceService.ALFRESCO_PREFIX + ":minimum",
String.valueOf(o.minimum));
}
if (o.maximum > 1)
{
//if 1 or unbounded -> no constraint
maxConstraint = "count(.) <= " + o.maximum;
bindElement.setAttributeNS(ALFRESCO_NS,
ALFRESCO_PREFIX + ":maximum",
bindElement.setAttributeNS(NamespaceService.ALFRESCO_URI,
NamespaceService.ALFRESCO_PREFIX + ":maximum",
String.valueOf(o.maximum));
}

View File

@@ -20,6 +20,7 @@ import java.io.*;
import javax.faces.context.FacesContext;
import javax.servlet.http.HttpServletRequest;
import org.alfresco.service.namespace.NamespaceService;
import org.alfresco.web.forms.*;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
@@ -28,6 +29,7 @@ import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.alfresco.web.app.servlet.FacesHelper;
import org.chiba.xml.ns.NamespaceConstants;
import org.chiba.xml.xforms.exception.XFormsException;
public class XFormsProcessor
@@ -36,6 +38,26 @@ public class XFormsProcessor
private static final Log LOGGER = LogFactory.getLog(XFormsProcessor.class);
private final static String[][] JS_NAMESPACES =
{
{ "xforms", NamespaceConstants.XFORMS_NS, NamespaceConstants.XFORMS_PREFIX },
{ "xhtml", NamespaceConstants.XHTML_NS, NamespaceConstants.XHTML_PREFIX },
{ "chiba", NamespaceConstants.CHIBA_NS, NamespaceConstants.CHIBA_PREFIX },
{ "alfresco", NamespaceService.ALFRESCO_URI, NamespaceService.ALFRESCO_PREFIX }
};
private final String[] JS_SCRIPTS =
{
"/scripts/tiny_mce/" + (LOGGER.isDebugEnabled()
? "tiny_mce_src.js"
: "tiny_mce.js"),
"/scripts/ajax/dojo/" + (LOGGER.isDebugEnabled()
? "dojo.js.uncompressed.js"
: "dojo.js"),
"/scripts/ajax/xforms.js"
};
public XFormsProcessor()
{
}
@@ -77,44 +99,51 @@ public class XFormsProcessor
throw new ProcessingException(xfe);
}
final String cp = fc.getExternalContext().getRequestContextPath();
final String contextPath = fc.getExternalContext().getRequestContextPath();
final Document result = XMLUtil.newDocument();
final String xformsUIDivId = "alfresco-xforms-ui";
// this div is where the ui will write to
final Element div = result.createElement("div");
div.setAttribute("id", "alfresco-xforms-ui");
div.setAttribute("id", xformsUIDivId);
result.appendChild(div);
Element e = result.createElement("link");
e.setAttribute("rel", "stylesheet");
e.setAttribute("type", "text/css");
e.setAttribute("href", cp + "/css/xforms.css");
e.setAttribute("href", contextPath + "/css/xforms.css");
div.appendChild(e);
// a script with config information and globals.
e = result.createElement("script");
e.setAttribute("type", "text/javascript");
e.appendChild(result.createTextNode("\ndjConfig = { isDebug: " + LOGGER.isDebugEnabled() +
" };\n" +
"var WEBAPP_CONTEXT = \"" + cp + "\";\n"));
final StringBuilder js = new StringBuilder("\ndjConfig = {isDebug:" + LOGGER.isDebugEnabled() + "};\n");
js.append("var alfresco_xforms_constants = {};\n");
js.append("alfresco_xforms_constants.WEBAPP_CONTEXT = '").
append(contextPath).
append("';\n");
js.append("alfresco_xforms_constants.XFORMS_UI_DIV_ID = '").
append(xformsUIDivId).
append("';\n");
for (String[] ns : JS_NAMESPACES)
{
js.append("alfresco_xforms_constants.").
append(ns[0].toUpperCase()).
append("_NS = '").append(ns[1]).append("';\n");
js.append("alfresco_xforms_constants.").
append(ns[0].toUpperCase()).
append("_PREFIX = '").append(ns[2]).append("';\n");
}
e.appendChild(result.createTextNode(js.toString()));
div.appendChild(e);
final String[] scripts =
{
"/scripts/tiny_mce/" + (LOGGER.isDebugEnabled()
? "tiny_mce_src.js"
: "tiny_mce.js"),
"/scripts/ajax/dojo/" + (LOGGER.isDebugEnabled()
? "dojo.js.uncompressed.js"
: "dojo.js"),
"/scripts/ajax/xforms.js"
};
// include all our scripts, order is significant
for (int i = 0; i < scripts.length; i++)
for (final String script : JS_SCRIPTS)
{
e = result.createElement("script");
e.setAttribute("type", "text/javascript");
e.setAttribute("src", cp + scripts[i]);
e.setAttribute("src", contextPath + script);
e.appendChild(result.createTextNode("\n"));
div.appendChild(e);
}