240 lines
7.6 KiB
C++
240 lines
7.6 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public
|
|
* License Version 1.1 (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.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
|
* implied. See the License for the specific language governing
|
|
* rights and limitations under the License.
|
|
*
|
|
* The Original Code is TransforMiiX XSLT processor.
|
|
*
|
|
* The Initial Developer of the Original Code is The MITRE Corporation.
|
|
* Portions created by MITRE are Copyright (C) 1999 The MITRE Corporation.
|
|
*
|
|
* Portions created by Keith Visco as a Non MITRE employee,
|
|
* (C) 1999 Keith Visco. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Keith Visco, kvisco@ziplink.net
|
|
* -- original author.
|
|
*
|
|
* Marina Mechtcheriakova, mmarina@mindspring.com
|
|
* -- changed some behavoir to be more compliant with spec
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* NodeSetFunctionCall
|
|
* A representation of the XPath NodeSet funtions
|
|
*/
|
|
|
|
#include "FunctionLib.h"
|
|
#include "XMLDOMUtils.h"
|
|
#include "Tokenizer.h"
|
|
#include "txAtom.h"
|
|
|
|
/*
|
|
* Creates a NodeSetFunctionCall of the given type
|
|
*/
|
|
NodeSetFunctionCall::NodeSetFunctionCall(NodeSetFunctions aType)
|
|
{
|
|
mType = aType;
|
|
switch (aType) {
|
|
case COUNT:
|
|
name = XPathNames::COUNT_FN;
|
|
break;
|
|
case ID:
|
|
name = XPathNames::ID_FN;
|
|
break;
|
|
case LAST:
|
|
name = XPathNames::LAST_FN;
|
|
break;
|
|
case LOCAL_NAME:
|
|
name = XPathNames::LOCAL_NAME_FN;
|
|
break;
|
|
case NAME:
|
|
name = XPathNames::NAME_FN;
|
|
break;
|
|
case NAMESPACE_URI:
|
|
name = XPathNames::NAMESPACE_URI_FN;
|
|
break;
|
|
case POSITION:
|
|
name = XPathNames::POSITION_FN;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Evaluates this Expr based on the given context node and processor state
|
|
* @param context the context node for evaluation of this Expr
|
|
* @param ps the ContextState containing the stack information needed
|
|
* for evaluation
|
|
* @return the result of the evaluation
|
|
*/
|
|
ExprResult* NodeSetFunctionCall::evaluate(Node* aContext, ContextState* aCs) {
|
|
ListIterator iter(¶ms);
|
|
switch (mType) {
|
|
case COUNT:
|
|
{
|
|
if (!requireParams(1, 1, aCs))
|
|
return new StringResult("error");
|
|
|
|
NodeSet* nodes;
|
|
nodes = evaluateToNodeSet((Expr*)iter.next(), aContext, aCs);
|
|
if (!nodes)
|
|
return new StringResult("error");
|
|
|
|
double count = nodes->size();
|
|
delete nodes;
|
|
return new NumberResult(count);
|
|
}
|
|
case ID:
|
|
{
|
|
if (!requireParams(1, 1, aCs))
|
|
return new StringResult("error");
|
|
|
|
ExprResult* exprResult;
|
|
exprResult = ((Expr*)iter.next())->evaluate(aContext, aCs);
|
|
if (!exprResult)
|
|
return new StringResult("error");
|
|
|
|
NodeSet* resultSet = new NodeSet();
|
|
if (!resultSet) {
|
|
// XXX ErrorReport: out of memory
|
|
return 0;
|
|
}
|
|
|
|
Document* contextDoc;
|
|
if (aContext->getNodeType() == Node::DOCUMENT_NODE)
|
|
contextDoc = (Document*)aContext;
|
|
else
|
|
contextDoc = aContext->getOwnerDocument();
|
|
|
|
if (exprResult->getResultType() == ExprResult::NODESET) {
|
|
NodeSet* nodes = (NodeSet*)exprResult;
|
|
int i;
|
|
for (i = 0; i < nodes->size(); i++) {
|
|
String idList, id;
|
|
XMLDOMUtils::getNodeValue(nodes->get(i), idList);
|
|
txTokenizer tokenizer(idList);
|
|
while (tokenizer.hasMoreTokens()) {
|
|
tokenizer.nextToken(id);
|
|
resultSet->add(contextDoc->getElementById(id));
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
String idList, id;
|
|
exprResult->stringValue(idList);
|
|
txTokenizer tokenizer(idList);
|
|
while (tokenizer.hasMoreTokens()) {
|
|
tokenizer.nextToken(id);
|
|
resultSet->add(contextDoc->getElementById(id));
|
|
}
|
|
}
|
|
delete exprResult;
|
|
|
|
return resultSet;
|
|
}
|
|
case LAST:
|
|
{
|
|
if (!requireParams(0, 0, aCs))
|
|
return new StringResult("error");
|
|
|
|
NodeSet* contextNodeSet = (NodeSet*)aCs->getNodeSetStack()->peek();
|
|
if (!contextNodeSet) {
|
|
String err("Internal error");
|
|
aCs->recieveError(err);
|
|
return new StringResult("error");
|
|
}
|
|
|
|
return new NumberResult(contextNodeSet->size());
|
|
}
|
|
case LOCAL_NAME:
|
|
case NAME:
|
|
case NAMESPACE_URI:
|
|
{
|
|
if (!requireParams(0, 1, aCs))
|
|
return new StringResult("error");
|
|
|
|
Node* node = 0;
|
|
// Check for optional arg
|
|
if (iter.hasNext()) {
|
|
NodeSet* nodes;
|
|
nodes = evaluateToNodeSet((Expr*)iter.next(), aContext, aCs);
|
|
if (!nodes)
|
|
return new StringResult("error");
|
|
|
|
if (nodes->isEmpty()) {
|
|
delete nodes;
|
|
return new StringResult();
|
|
}
|
|
node = nodes->get(0);
|
|
delete nodes;
|
|
}
|
|
else {
|
|
node = aContext;
|
|
}
|
|
|
|
switch (mType) {
|
|
case LOCAL_NAME:
|
|
{
|
|
String localName;
|
|
txAtom* localNameAtom;
|
|
node->getLocalName(&localNameAtom);
|
|
if (localNameAtom) {
|
|
// Node has a localName
|
|
TX_GET_ATOM_STRING(localNameAtom, localName);
|
|
TX_RELEASE_ATOM(localNameAtom);
|
|
}
|
|
|
|
return new StringResult(localName);
|
|
}
|
|
case NAMESPACE_URI:
|
|
{
|
|
return new StringResult(node->getNamespaceURI());
|
|
}
|
|
case NAME:
|
|
{
|
|
switch (node->getNodeType()) {
|
|
case Node::ATTRIBUTE_NODE:
|
|
case Node::ELEMENT_NODE:
|
|
case Node::PROCESSING_INSTRUCTION_NODE:
|
|
// XXX Namespace: namespaces have a name
|
|
return new StringResult(node->getNodeName());
|
|
default:
|
|
break;
|
|
}
|
|
return new StringResult();
|
|
}
|
|
default:
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
case POSITION:
|
|
{
|
|
if (!requireParams(0, 0, aCs))
|
|
return new StringResult("error");
|
|
|
|
NodeSet* contextNodeSet = (NodeSet*)aCs->getNodeSetStack()->peek();
|
|
if (!contextNodeSet) {
|
|
String err("Internal error");
|
|
aCs->recieveError(err);
|
|
return new StringResult("error");
|
|
}
|
|
|
|
return new NumberResult(contextNodeSet->indexOf(aContext) + 1);
|
|
}
|
|
}
|
|
|
|
String err("Internal error");
|
|
aCs->recieveError(err);
|
|
return new StringResult("error");
|
|
}
|