org.orbeon.oxf.xforms.action.actions.XFormsDeleteAction.java Source code

Java tutorial

Introduction

Here is the source code for org.orbeon.oxf.xforms.action.actions.XFormsDeleteAction.java

Source

/**
 * Copyright (C) 2010 Orbeon, Inc.
 *
 * This program is free software; you can redistribute it and/or modify it under the terms of the
 * GNU Lesser General Public License as published by the Free Software Foundation; either version
 * 2.1 of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * See the GNU Lesser General Public License for more details.
 *
 * The full text of the license is available at http://www.gnu.org/copyleft/lesser.html
 */
package org.orbeon.oxf.xforms.action.actions;

import org.dom4j.Attribute;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.Node;
import org.orbeon.oxf.util.IndentedLogger;
import org.orbeon.oxf.xforms.*;
import org.orbeon.oxf.xforms.action.XFormsAction;
import org.orbeon.oxf.xforms.action.XFormsActionInterpreter;
import org.orbeon.oxf.xforms.event.Dispatch;
import org.orbeon.oxf.xforms.event.events.XFormsDeleteEvent;
import org.orbeon.oxf.xforms.xbl.Scope;
import org.orbeon.saxon.om.Item;
import org.orbeon.saxon.om.NodeInfo;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

/**
 * 9.3.6 The delete Element
 */
public class XFormsDeleteAction extends XFormsAction {

    public static final String CANNOT_DELETE_READONLY_MESSAGE = "Cannot perform deletion in read-only instance.";

    public void execute(XFormsActionInterpreter actionInterpreter, Element actionElement, Scope actionScope,
            boolean hasOverriddenContext, Item overriddenContext) {

        final IndentedLogger indentedLogger = actionInterpreter.indentedLogger();
        final XFormsContainingDocument containingDocument = actionInterpreter.containingDocument();
        final XFormsContextStack contextStack = actionInterpreter.actionXPathContext();

        final String atAttribute = actionElement.attributeValue("at");
        final String contextAttribute = actionElement.attributeValue(XFormsConstants.CONTEXT_QNAME);

        final List<Item> collectionToUpdate = contextStack.getCurrentNodeset();
        final boolean isEmptyNodesetBinding = collectionToUpdate == null || collectionToUpdate.size() == 0;

        // "The delete action is terminated with no effect if [...] the context attribute is not given and the Node
        // Set Binding node-set is empty."
        if (contextAttribute == null && isEmptyNodesetBinding) {
            if (indentedLogger.isDebugEnabled())
                indentedLogger.logDebug("xf:delete", "context is empty, terminating");
            return;
        }

        // Handle insert context (with @context attribute)
        // "The delete action is terminated with no effect if the insert context is the empty node-set [...]."
        if (hasOverriddenContext && (overriddenContext == null || !(overriddenContext instanceof NodeInfo))) {
            if (indentedLogger.isDebugEnabled())
                indentedLogger.logDebug("xf:delete",
                        "overridden context is an empty nodeset or not a nodeset, terminating");
            return;
        }

        {
            int deleteIndex;
            {
                if (isEmptyNodesetBinding) {
                    // "If the Node Set Binding node-set empty, then this attribute is ignored"
                    deleteIndex = 0;
                } else if (atAttribute == null) {
                    // "If there is no delete location, each node in the Node Set Binding node-set is deleted, except
                    // if the node is a readonly node or the root document element of an instance then that particular
                    // node is not deleted."

                    deleteIndex = -1;
                } else {
                    // "1. The evaluation context node is the first node in document order from the Node Set Binding
                    // node-set, the context size is the size of the Node Set Binding node-set, and the context
                    // position is 1."

                    // "2. The return value is processed according to the rules of the XPath function round()"
                    final String insertionIndexString = actionInterpreter.evaluateAsString(actionElement,
                            collectionToUpdate, 1, "round(" + atAttribute + ")");

                    // "3. If the result is in the range 1 to the Node Set Binding node-set size, then the insert
                    // location is equal to the result. If the result is non-positive, then the insert location is
                    // 1. Otherwise, the result is NaN or exceeds the Node Set Binding node-set size, so the insert
                    // location is the Node Set Binding node-set size."

                    // Don't think we will get NaN with XPath 2.0...
                    deleteIndex = "NaN".equals(insertionIndexString) ? collectionToUpdate.size()
                            : Integer.parseInt(insertionIndexString);

                    // Adjust index to be in range
                    if (deleteIndex > collectionToUpdate.size())
                        deleteIndex = collectionToUpdate.size();

                    if (deleteIndex < 1)
                        deleteIndex = 1;
                }
            }

            doDelete(containingDocument, indentedLogger, collectionToUpdate, deleteIndex, true);
        }
    }

    public static class DeleteInfo {
        public final NodeInfo parent;
        public final NodeInfo nodeInfo;
        public final int index;

        private DeleteInfo(NodeInfo parent, NodeInfo nodeInfo, int index) {
            this.parent = parent;
            this.nodeInfo = nodeInfo;
            this.index = index; // only really makes sense for element content, not attribute content
        }
    }

    public static List<DeleteInfo> doDelete(XFormsContainingDocument containingDocument,
            IndentedLogger indentedLogger, List collectionToUpdate, int deleteIndex, boolean doDispatch) {

        final boolean isEmptyNodesetBinding = collectionToUpdate == null || collectionToUpdate.size() == 0;

        final List<DeleteInfo> deleteInfos;
        if (isEmptyNodesetBinding) {
            deleteInfos = Collections.emptyList();
        } else if (deleteIndex == -1) {
            // Delete the entire collection

            deleteInfos = new ArrayList<DeleteInfo>(collectionToUpdate.size());
            for (int i = 1; i <= collectionToUpdate.size(); i++) {
                final DeleteInfo deletedInfo = doDeleteOne(indentedLogger, collectionToUpdate, i);
                if (deletedInfo != null) {
                    deleteInfos.add(deletedInfo);
                }
            }
        } else {
            // Find actual deletion point

            final DeleteInfo deleteInfo = doDeleteOne(indentedLogger, collectionToUpdate, deleteIndex);
            if (deleteInfo != null) {
                deleteInfos = Collections.singletonList(deleteInfo);
            } else {
                deleteInfos = Collections.emptyList();
            }
        }

        if (deleteInfos.size() == 0) {
            if (indentedLogger != null && indentedLogger.isDebugEnabled())
                indentedLogger.logDebug("xf:delete", "empty collection, terminating");
        } else if (containingDocument != null) {
            // Identify the instance that actually changes
            // NOTE: More than one instance may be modified. For now we look at the first one.
            final XFormsInstance modifiedInstance = containingDocument
                    .getInstanceForNode(deleteInfos.get(0).nodeInfo);

            if (indentedLogger != null && indentedLogger.isDebugEnabled())
                indentedLogger.logDebug("xf:delete", "removed nodes", "count", Integer.toString(deleteInfos.size()),
                        "instance", (modifiedInstance != null) ? modifiedInstance.getEffectiveId() : null);

            if (modifiedInstance != null) {
                // NOTE: Can be null if document into which delete is performed is not in an instance, e.g. in a variable

                // "XForms Actions that change the tree structure of instance data result in setting all four flags to true"
                modifiedInstance.markModified();
                modifiedInstance.model().markStructuralChange(modifiedInstance);

                // "4. If the delete is successful, the event xforms-delete is dispatched."
                if (doDispatch)
                    Dispatch.dispatchEvent(new XFormsDeleteEvent(modifiedInstance, deleteInfos, deleteIndex));
            }
        }

        return deleteInfos;
    }

    private static DeleteInfo doDeleteOne(IndentedLogger indentedLogger, List collectionToUpdate, int deleteIndex) {
        final NodeInfo nodeInfoToRemove = (NodeInfo) collectionToUpdate.get(deleteIndex - 1);
        final NodeInfo parentNodeInfo = nodeInfoToRemove.getParent();

        final Node nodeToRemove = XFormsUtils.getNodeFromNodeInfo(nodeInfoToRemove, CANNOT_DELETE_READONLY_MESSAGE);

        final List contentToUpdate;
        final int indexInContentToUpdate;
        final Element parentElement = nodeToRemove.getParent();
        if (parentElement != null) {
            // Regular case
            if (nodeToRemove instanceof Attribute) {
                contentToUpdate = parentElement.attributes();
            } else {
                contentToUpdate = parentElement.content();
            }
            indexInContentToUpdate = contentToUpdate.indexOf(nodeToRemove);
        } else if (nodeToRemove.getDocument() != null
                && nodeToRemove == nodeToRemove.getDocument().getRootElement()) {
            // Case of root element where parent is Document
            contentToUpdate = nodeToRemove.getDocument().content();
            indexInContentToUpdate = contentToUpdate.indexOf(nodeToRemove);
        } else if (nodeToRemove instanceof Document) {
            // Case where node to remove is Document

            // "except if the node is the root document element of an instance then the delete action
            // is terminated with no effect."

            if (indentedLogger.isDebugEnabled())
                indentedLogger.logDebug("xf:delete", "ignoring attempt to delete document node");

            return null;
        } else {
            // Node to remove doesn't have a parent so we can't delete it
            // This can happen for nodes already detached, or nodes newly created with e.g. xf:element()
            return null;
        }

        // Actually perform the deletion
        // "The node at the delete location in the Node Set Binding node-set is deleted"
        contentToUpdate.remove(indexInContentToUpdate);

        return new DeleteInfo(parentNodeInfo, nodeInfoToRemove, indexInContentToUpdate);
    }
}