1*cdf0e10cSrcweir /************************************************************************* 2*cdf0e10cSrcweir * 3*cdf0e10cSrcweir * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4*cdf0e10cSrcweir * 5*cdf0e10cSrcweir * Copyright 2000, 2010 Oracle and/or its affiliates. 6*cdf0e10cSrcweir * 7*cdf0e10cSrcweir * OpenOffice.org - a multi-platform office productivity suite 8*cdf0e10cSrcweir * 9*cdf0e10cSrcweir * This file is part of OpenOffice.org. 10*cdf0e10cSrcweir * 11*cdf0e10cSrcweir * OpenOffice.org is free software: you can redistribute it and/or modify 12*cdf0e10cSrcweir * it under the terms of the GNU Lesser General Public License version 3 13*cdf0e10cSrcweir * only, as published by the Free Software Foundation. 14*cdf0e10cSrcweir * 15*cdf0e10cSrcweir * OpenOffice.org is distributed in the hope that it will be useful, 16*cdf0e10cSrcweir * but WITHOUT ANY WARRANTY; without even the implied warranty of 17*cdf0e10cSrcweir * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 18*cdf0e10cSrcweir * GNU Lesser General Public License version 3 for more details 19*cdf0e10cSrcweir * (a copy is included in the LICENSE file that accompanied this code). 20*cdf0e10cSrcweir * 21*cdf0e10cSrcweir * You should have received a copy of the GNU Lesser General Public License 22*cdf0e10cSrcweir * version 3 along with OpenOffice.org. If not, see 23*cdf0e10cSrcweir * <http://www.openoffice.org/license.html> 24*cdf0e10cSrcweir * for a copy of the LGPLv3 License. 25*cdf0e10cSrcweir * 26*cdf0e10cSrcweir ************************************************************************/ 27*cdf0e10cSrcweir 28*cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove 29*cdf0e10cSrcweir #include "precompiled_forms.hxx" 30*cdf0e10cSrcweir 31*cdf0e10cSrcweir #include "model.hxx" 32*cdf0e10cSrcweir #include "model_helper.hxx" 33*cdf0e10cSrcweir #include "mip.hxx" 34*cdf0e10cSrcweir #include "evaluationcontext.hxx" 35*cdf0e10cSrcweir #include "unohelper.hxx" 36*cdf0e10cSrcweir #include "submission/serialization_app_xml.hxx" 37*cdf0e10cSrcweir #include "resourcehelper.hxx" 38*cdf0e10cSrcweir #include "xmlhelper.hxx" 39*cdf0e10cSrcweir #include "convert.hxx" 40*cdf0e10cSrcweir 41*cdf0e10cSrcweir #include <rtl/ustring.hxx> 42*cdf0e10cSrcweir #include <rtl/ustrbuf.hxx> 43*cdf0e10cSrcweir #include <tools/debug.hxx> 44*cdf0e10cSrcweir 45*cdf0e10cSrcweir // UNO classes 46*cdf0e10cSrcweir #include <com/sun/star/xml/dom/XNode.hpp> 47*cdf0e10cSrcweir #include <com/sun/star/xml/dom/XDocumentBuilder.hpp> 48*cdf0e10cSrcweir #include <com/sun/star/xml/dom/XDocumentFragment.hpp> 49*cdf0e10cSrcweir #include <com/sun/star/xml/dom/XNamedNodeMap.hpp> 50*cdf0e10cSrcweir #include <com/sun/star/xml/xpath/XXPathObject.hpp> 51*cdf0e10cSrcweir #include <com/sun/star/xml/xpath/XPathObjectType.hpp> 52*cdf0e10cSrcweir #include <com/sun/star/beans/PropertyValue.hpp> 53*cdf0e10cSrcweir #include <com/sun/star/io/XInputStream.hpp> 54*cdf0e10cSrcweir #include <com/sun/star/io/XActiveDataSink.hpp> 55*cdf0e10cSrcweir #include <com/sun/star/io/XTextInputStream.hpp> 56*cdf0e10cSrcweir #include <com/sun/star/container/XEnumeration.hpp> 57*cdf0e10cSrcweir #include <com/sun/star/container/XNameContainer.hpp> 58*cdf0e10cSrcweir #include <com/sun/star/frame/XModel.hpp> 59*cdf0e10cSrcweir #include <com/sun/star/xforms/XFormsSupplier.hpp> 60*cdf0e10cSrcweir #include <com/sun/star/xforms/XDataTypeRepository.hpp> 61*cdf0e10cSrcweir #include <com/sun/star/xsd/XDataType.hpp> 62*cdf0e10cSrcweir #include <com/sun/star/xsd/DataTypeClass.hpp> 63*cdf0e10cSrcweir 64*cdf0e10cSrcweir 65*cdf0e10cSrcweir using rtl::OUString; 66*cdf0e10cSrcweir using rtl::OUStringBuffer; 67*cdf0e10cSrcweir using com::sun::star::beans::PropertyValue; 68*cdf0e10cSrcweir using com::sun::star::io::XInputStream; 69*cdf0e10cSrcweir using com::sun::star::io::XActiveDataSink; 70*cdf0e10cSrcweir using com::sun::star::io::XTextInputStream; 71*cdf0e10cSrcweir using com::sun::star::container::XEnumeration; 72*cdf0e10cSrcweir using com::sun::star::container::XNameContainer; 73*cdf0e10cSrcweir using com::sun::star::xforms::XFormsSupplier; 74*cdf0e10cSrcweir 75*cdf0e10cSrcweir using namespace xforms; 76*cdf0e10cSrcweir using namespace com::sun::star::uno; 77*cdf0e10cSrcweir using namespace com::sun::star::xml::dom; 78*cdf0e10cSrcweir using namespace com::sun::star::xml::xpath; 79*cdf0e10cSrcweir 80*cdf0e10cSrcweir 81*cdf0e10cSrcweir 82*cdf0e10cSrcweir // 83*cdf0e10cSrcweir // implement XFormsUIHelper1 84*cdf0e10cSrcweir // 85*cdf0e10cSrcweir 86*cdf0e10cSrcweir OUString Model::getDefaultServiceNameForNode( const XNode_t& xNode ) 87*cdf0e10cSrcweir throw( RuntimeException ) 88*cdf0e10cSrcweir { 89*cdf0e10cSrcweir // determine service for control. string/text field is default. 90*cdf0e10cSrcweir OUString sService = OUSTRING("com.sun.star.form.component.TextField"); 91*cdf0e10cSrcweir 92*cdf0e10cSrcweir // query repository for suitable type 93*cdf0e10cSrcweir OSL_ENSURE( mxDataTypes.is(), "no type repository?" ); 94*cdf0e10cSrcweir OUString sTypeName = queryMIP( xNode ).getTypeName(); 95*cdf0e10cSrcweir if( mxDataTypes->hasByName( sTypeName ) ) 96*cdf0e10cSrcweir { 97*cdf0e10cSrcweir OSL_ENSURE( mxDataTypes->getDataType( sTypeName ).is(), 98*cdf0e10cSrcweir "has or has not?" ); 99*cdf0e10cSrcweir 100*cdf0e10cSrcweir switch( mxDataTypes->getDataType( sTypeName )->getTypeClass() ) 101*cdf0e10cSrcweir { 102*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::BOOLEAN: 103*cdf0e10cSrcweir sService = OUSTRING("com.sun.star.form.component.CheckBox"); 104*cdf0e10cSrcweir break; 105*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::DOUBLE: 106*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::DECIMAL: 107*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::FLOAT: 108*cdf0e10cSrcweir sService = OUSTRING("com.sun.star.form.component.NumericField"); 109*cdf0e10cSrcweir break; 110*cdf0e10cSrcweir 111*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::STRING: 112*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::DURATION: 113*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::DATETIME: 114*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::TIME: 115*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::DATE: 116*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::gYearMonth: 117*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::gYear: 118*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::gMonthDay: 119*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::gDay: 120*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::gMonth: 121*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::hexBinary: 122*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::base64Binary: 123*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::anyURI: 124*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::QName: 125*cdf0e10cSrcweir case com::sun::star::xsd::DataTypeClass::NOTATION: 126*cdf0e10cSrcweir default: 127*cdf0e10cSrcweir // keep default 128*cdf0e10cSrcweir break; 129*cdf0e10cSrcweir } 130*cdf0e10cSrcweir } 131*cdf0e10cSrcweir 132*cdf0e10cSrcweir return sService; 133*cdf0e10cSrcweir } 134*cdf0e10cSrcweir 135*cdf0e10cSrcweir 136*cdf0e10cSrcweir void lcl_OutPosition( OUStringBuffer& rBuffer, 137*cdf0e10cSrcweir const Reference<XNode>& xNode ) 138*cdf0e10cSrcweir { 139*cdf0e10cSrcweir OSL_ENSURE( xNode->getParentNode().is(), "need parent" ); 140*cdf0e10cSrcweir 141*cdf0e10cSrcweir // count # of occurences of this node 142*cdf0e10cSrcweir sal_Int32 nFound = 0; 143*cdf0e10cSrcweir sal_Int32 nPosition = -1; 144*cdf0e10cSrcweir if( xNode->getParentNode().is() ) 145*cdf0e10cSrcweir { 146*cdf0e10cSrcweir for( Reference<XNode> xIter = xNode->getParentNode()->getFirstChild(); 147*cdf0e10cSrcweir xIter != NULL; 148*cdf0e10cSrcweir xIter = xIter->getNextSibling() ) 149*cdf0e10cSrcweir { 150*cdf0e10cSrcweir if( xIter->getNodeType() == xNode->getNodeType() && 151*cdf0e10cSrcweir xIter->getNodeName() == xNode->getNodeName() && 152*cdf0e10cSrcweir xIter->getNamespaceURI() == xNode->getNamespaceURI() ) 153*cdf0e10cSrcweir { 154*cdf0e10cSrcweir nFound++; 155*cdf0e10cSrcweir if( xIter == xNode ) 156*cdf0e10cSrcweir nPosition = nFound; 157*cdf0e10cSrcweir } 158*cdf0e10cSrcweir } 159*cdf0e10cSrcweir } 160*cdf0e10cSrcweir OSL_ENSURE( nFound > 0 && nPosition > 0, "node not found???" ); 161*cdf0e10cSrcweir 162*cdf0e10cSrcweir // output position (if necessary) 163*cdf0e10cSrcweir if( nFound > 1 ) 164*cdf0e10cSrcweir { 165*cdf0e10cSrcweir rBuffer.insert( 0, sal_Unicode(']') ); 166*cdf0e10cSrcweir rBuffer.insert( 0, nPosition ); 167*cdf0e10cSrcweir rBuffer.insert( 0, sal_Unicode('[') ); 168*cdf0e10cSrcweir } 169*cdf0e10cSrcweir } 170*cdf0e10cSrcweir 171*cdf0e10cSrcweir void lcl_OutName( OUStringBuffer& rBuffer, 172*cdf0e10cSrcweir const Reference<XNode>& xNode ) 173*cdf0e10cSrcweir { 174*cdf0e10cSrcweir rBuffer.insert( 0, xNode->getNodeName() ); 175*cdf0e10cSrcweir OUString sPrefix = xNode->getPrefix(); 176*cdf0e10cSrcweir if( sPrefix.getLength() > 0 ) 177*cdf0e10cSrcweir { 178*cdf0e10cSrcweir rBuffer.insert( 0, sal_Unicode(':') ); 179*cdf0e10cSrcweir rBuffer.insert( 0, sPrefix ); 180*cdf0e10cSrcweir } 181*cdf0e10cSrcweir } 182*cdf0e10cSrcweir 183*cdf0e10cSrcweir void lcl_OutInstance( OUStringBuffer& rBuffer, 184*cdf0e10cSrcweir const Reference<XNode>& xNode, 185*cdf0e10cSrcweir Model* pModel ) 186*cdf0e10cSrcweir { 187*cdf0e10cSrcweir Reference<XDocument> xDoc = xNode->getOwnerDocument(); 188*cdf0e10cSrcweir 189*cdf0e10cSrcweir if( xDoc != pModel->getDefaultInstance() ) 190*cdf0e10cSrcweir { 191*cdf0e10cSrcweir rBuffer.insert( 0, OUSTRING("')") ); 192*cdf0e10cSrcweir 193*cdf0e10cSrcweir // iterate over instances, and find the right one 194*cdf0e10cSrcweir OUString sInstanceName; 195*cdf0e10cSrcweir Reference<XEnumeration> xEnum = 196*cdf0e10cSrcweir pModel->getInstances()->createEnumeration(); 197*cdf0e10cSrcweir while( ( sInstanceName.getLength() == 0 ) && xEnum->hasMoreElements() ) 198*cdf0e10cSrcweir { 199*cdf0e10cSrcweir Sequence<PropertyValue> aValues; 200*cdf0e10cSrcweir xEnum->nextElement() >>= aValues; 201*cdf0e10cSrcweir 202*cdf0e10cSrcweir // get ID and instance 203*cdf0e10cSrcweir OUString sId; 204*cdf0e10cSrcweir Reference<XDocument> xInstance; 205*cdf0e10cSrcweir getInstanceData( aValues, &sId, &xInstance, NULL, NULL ); 206*cdf0e10cSrcweir 207*cdf0e10cSrcweir // now check whether this was our instance: 208*cdf0e10cSrcweir if( xInstance == xDoc ) 209*cdf0e10cSrcweir sInstanceName = sId; 210*cdf0e10cSrcweir } 211*cdf0e10cSrcweir 212*cdf0e10cSrcweir rBuffer.insert( 0, sInstanceName ); 213*cdf0e10cSrcweir rBuffer.insert( 0, OUSTRING("instance('") ); 214*cdf0e10cSrcweir } 215*cdf0e10cSrcweir } 216*cdf0e10cSrcweir 217*cdf0e10cSrcweir OUString Model::getDefaultBindingExpressionForNode( 218*cdf0e10cSrcweir const XNode_t& xNode, 219*cdf0e10cSrcweir const EvaluationContext& rContext) 220*cdf0e10cSrcweir { 221*cdf0e10cSrcweir OSL_ENSURE( xNode.is(), "need node" ); 222*cdf0e10cSrcweir 223*cdf0e10cSrcweir // iterate upwards and put sections into the expression buffer. 224*cdf0e10cSrcweir // Stop iteration either at context node (relative expression) or 225*cdf0e10cSrcweir // at document root, whichever occurs first. 226*cdf0e10cSrcweir OUStringBuffer aBuffer; 227*cdf0e10cSrcweir for( Reference<XNode> xCurrent = xNode; 228*cdf0e10cSrcweir xCurrent.is() && xCurrent != rContext.mxContextNode; 229*cdf0e10cSrcweir xCurrent = xCurrent->getParentNode() ) 230*cdf0e10cSrcweir { 231*cdf0e10cSrcweir // insert a '/' for every step except the first 232*cdf0e10cSrcweir if( aBuffer.getLength() > 0 ) 233*cdf0e10cSrcweir aBuffer.insert( 0, sal_Unicode('/') ); 234*cdf0e10cSrcweir 235*cdf0e10cSrcweir switch( xCurrent->getNodeType() ) 236*cdf0e10cSrcweir { 237*cdf0e10cSrcweir case NodeType_ELEMENT_NODE: 238*cdf0e10cSrcweir lcl_OutPosition( aBuffer, xCurrent ); 239*cdf0e10cSrcweir lcl_OutName( aBuffer, xCurrent ); 240*cdf0e10cSrcweir break; 241*cdf0e10cSrcweir 242*cdf0e10cSrcweir case NodeType_TEXT_NODE: 243*cdf0e10cSrcweir lcl_OutPosition( aBuffer, xCurrent ); 244*cdf0e10cSrcweir aBuffer.insert( 0, OUSTRING("text()") ); 245*cdf0e10cSrcweir break; 246*cdf0e10cSrcweir 247*cdf0e10cSrcweir case NodeType_ATTRIBUTE_NODE: 248*cdf0e10cSrcweir lcl_OutName( aBuffer, xCurrent ); 249*cdf0e10cSrcweir aBuffer.insert( 0, sal_Unicode('@') ); 250*cdf0e10cSrcweir break; 251*cdf0e10cSrcweir 252*cdf0e10cSrcweir case NodeType_DOCUMENT_NODE: 253*cdf0e10cSrcweir // check for which instance we have 254*cdf0e10cSrcweir lcl_OutInstance( aBuffer, xCurrent, this ); 255*cdf0e10cSrcweir break; 256*cdf0e10cSrcweir 257*cdf0e10cSrcweir default: 258*cdf0e10cSrcweir // unknown type? fail! 259*cdf0e10cSrcweir OSL_ENSURE( false, "unknown node type!" ); 260*cdf0e10cSrcweir xCurrent.set( NULL ); 261*cdf0e10cSrcweir aBuffer.makeStringAndClear(); 262*cdf0e10cSrcweir // we'll remove the slash below 263*cdf0e10cSrcweir aBuffer.insert( 0, sal_Unicode('/') ); 264*cdf0e10cSrcweir break; 265*cdf0e10cSrcweir } 266*cdf0e10cSrcweir } 267*cdf0e10cSrcweir 268*cdf0e10cSrcweir return aBuffer.makeStringAndClear(); 269*cdf0e10cSrcweir } 270*cdf0e10cSrcweir 271*cdf0e10cSrcweir 272*cdf0e10cSrcweir 273*cdf0e10cSrcweir OUString Model::getDefaultBindingExpressionForNode( const XNode_t& xNode ) 274*cdf0e10cSrcweir throw( RuntimeException ) 275*cdf0e10cSrcweir { 276*cdf0e10cSrcweir return getDefaultBindingExpressionForNode( xNode, getEvaluationContext() ); 277*cdf0e10cSrcweir } 278*cdf0e10cSrcweir 279*cdf0e10cSrcweir bool lcl_isWhitespace( const OUString& rString ) 280*cdf0e10cSrcweir { 281*cdf0e10cSrcweir sal_Int32 nLength = rString.getLength(); 282*cdf0e10cSrcweir const sal_Unicode* pStr = rString.getStr(); 283*cdf0e10cSrcweir 284*cdf0e10cSrcweir bool bWhitespace = true; 285*cdf0e10cSrcweir for( sal_Int32 i = 0; bWhitespace && ( i < nLength ); i++ ) 286*cdf0e10cSrcweir { 287*cdf0e10cSrcweir sal_Unicode c = pStr[i]; 288*cdf0e10cSrcweir bWhitespace = ( c == sal_Unicode(0x09) || 289*cdf0e10cSrcweir c == sal_Unicode(0x0A) || 290*cdf0e10cSrcweir c == sal_Unicode(0x0D) || 291*cdf0e10cSrcweir c == sal_Unicode(0x20) ); 292*cdf0e10cSrcweir } 293*cdf0e10cSrcweir return bWhitespace; 294*cdf0e10cSrcweir } 295*cdf0e10cSrcweir 296*cdf0e10cSrcweir OUString Model::getNodeDisplayName( const XNode_t& xNode, 297*cdf0e10cSrcweir sal_Bool bDetail ) 298*cdf0e10cSrcweir throw( RuntimeException ) 299*cdf0e10cSrcweir { 300*cdf0e10cSrcweir OUStringBuffer aBuffer; 301*cdf0e10cSrcweir 302*cdf0e10cSrcweir switch( xNode->getNodeType() ) 303*cdf0e10cSrcweir { 304*cdf0e10cSrcweir case NodeType_ELEMENT_NODE: 305*cdf0e10cSrcweir lcl_OutName( aBuffer, xNode ); 306*cdf0e10cSrcweir break; 307*cdf0e10cSrcweir 308*cdf0e10cSrcweir case NodeType_TEXT_NODE: 309*cdf0e10cSrcweir { 310*cdf0e10cSrcweir OUString sContent = xNode->getNodeValue(); 311*cdf0e10cSrcweir if( bDetail || ! lcl_isWhitespace( sContent ) ) 312*cdf0e10cSrcweir { 313*cdf0e10cSrcweir aBuffer.append( sal_Unicode('"') ); 314*cdf0e10cSrcweir aBuffer.append( Convert::collapseWhitespace( sContent ) ); 315*cdf0e10cSrcweir aBuffer.append( sal_Unicode('"') ); 316*cdf0e10cSrcweir } 317*cdf0e10cSrcweir } 318*cdf0e10cSrcweir break; 319*cdf0e10cSrcweir 320*cdf0e10cSrcweir case NodeType_ATTRIBUTE_NODE: 321*cdf0e10cSrcweir lcl_OutName( aBuffer, xNode ); 322*cdf0e10cSrcweir aBuffer.insert( 0, sal_Unicode('@') ); 323*cdf0e10cSrcweir break; 324*cdf0e10cSrcweir 325*cdf0e10cSrcweir case NodeType_DOCUMENT_NODE: 326*cdf0e10cSrcweir if( xNode == getDefaultInstance() ) 327*cdf0e10cSrcweir aBuffer.append( sal_Unicode('/') ); 328*cdf0e10cSrcweir else 329*cdf0e10cSrcweir lcl_OutInstance( aBuffer, xNode, this ); 330*cdf0e10cSrcweir break; 331*cdf0e10cSrcweir 332*cdf0e10cSrcweir default: 333*cdf0e10cSrcweir // unknown type? fail! 334*cdf0e10cSrcweir OSL_ENSURE( false, "unknown node type!" ); 335*cdf0e10cSrcweir break; 336*cdf0e10cSrcweir } 337*cdf0e10cSrcweir 338*cdf0e10cSrcweir return aBuffer.makeStringAndClear(); 339*cdf0e10cSrcweir } 340*cdf0e10cSrcweir 341*cdf0e10cSrcweir OUString Model::getNodeName( const XNode_t& xNode ) 342*cdf0e10cSrcweir throw( RuntimeException ) 343*cdf0e10cSrcweir { 344*cdf0e10cSrcweir OUStringBuffer aBuffer; 345*cdf0e10cSrcweir 346*cdf0e10cSrcweir switch( xNode->getNodeType() ) 347*cdf0e10cSrcweir { 348*cdf0e10cSrcweir case NodeType_ELEMENT_NODE: 349*cdf0e10cSrcweir case NodeType_ATTRIBUTE_NODE: 350*cdf0e10cSrcweir lcl_OutName( aBuffer, xNode ); 351*cdf0e10cSrcweir break; 352*cdf0e10cSrcweir 353*cdf0e10cSrcweir case NodeType_TEXT_NODE: 354*cdf0e10cSrcweir case NodeType_DOCUMENT_NODE: 355*cdf0e10cSrcweir default: 356*cdf0e10cSrcweir // unknown type? fail! 357*cdf0e10cSrcweir OSL_ENSURE( false, "no name for this node type!" ); 358*cdf0e10cSrcweir break; 359*cdf0e10cSrcweir } 360*cdf0e10cSrcweir 361*cdf0e10cSrcweir return aBuffer.makeStringAndClear(); 362*cdf0e10cSrcweir } 363*cdf0e10cSrcweir 364*cdf0e10cSrcweir OUString Model::getBindingName( const XPropertySet_t& xBinding, 365*cdf0e10cSrcweir sal_Bool /*bDetail*/ ) 366*cdf0e10cSrcweir throw( RuntimeException ) 367*cdf0e10cSrcweir { 368*cdf0e10cSrcweir OUString sID; 369*cdf0e10cSrcweir xBinding->getPropertyValue( OUSTRING("BindingID" ) ) >>= sID; 370*cdf0e10cSrcweir OUString sExpression; 371*cdf0e10cSrcweir xBinding->getPropertyValue( OUSTRING("BindingExpression" ) ) >>= sExpression; 372*cdf0e10cSrcweir 373*cdf0e10cSrcweir OUStringBuffer aBuffer; 374*cdf0e10cSrcweir if( sID.getLength() > 0 ) 375*cdf0e10cSrcweir { 376*cdf0e10cSrcweir aBuffer.append( sID ); 377*cdf0e10cSrcweir aBuffer.append( OUSTRING(" (" )); 378*cdf0e10cSrcweir aBuffer.append( sExpression ); 379*cdf0e10cSrcweir aBuffer.append( OUSTRING(")" )); 380*cdf0e10cSrcweir } 381*cdf0e10cSrcweir else 382*cdf0e10cSrcweir aBuffer.append( sExpression ); 383*cdf0e10cSrcweir 384*cdf0e10cSrcweir return aBuffer.makeStringAndClear(); 385*cdf0e10cSrcweir } 386*cdf0e10cSrcweir 387*cdf0e10cSrcweir OUString Model::getSubmissionName( const XPropertySet_t& xSubmission, 388*cdf0e10cSrcweir sal_Bool /*bDetail*/ ) 389*cdf0e10cSrcweir throw( RuntimeException ) 390*cdf0e10cSrcweir { 391*cdf0e10cSrcweir OUString sID; 392*cdf0e10cSrcweir xSubmission->getPropertyValue( OUSTRING("ID") ) >>= sID; 393*cdf0e10cSrcweir return sID; 394*cdf0e10cSrcweir } 395*cdf0e10cSrcweir 396*cdf0e10cSrcweir Model::XPropertySet_t Model::cloneBindingAsGhost( const XPropertySet_t &xBinding ) 397*cdf0e10cSrcweir throw( RuntimeException ) 398*cdf0e10cSrcweir { 399*cdf0e10cSrcweir // Create a new binding instance first... 400*cdf0e10cSrcweir Binding *pBinding = new Binding(); 401*cdf0e10cSrcweir 402*cdf0e10cSrcweir // ...and bump up the "defered notification counter" 403*cdf0e10cSrcweir // to prevent this binding from contributing to the 404*cdf0e10cSrcweir // MIPs table... 405*cdf0e10cSrcweir pBinding->deferNotifications(true); 406*cdf0e10cSrcweir 407*cdf0e10cSrcweir // Copy the propertyset and return result... 408*cdf0e10cSrcweir XPropertySet_t xNewBinding(pBinding); 409*cdf0e10cSrcweir copy( xBinding, xNewBinding ); 410*cdf0e10cSrcweir return xNewBinding; 411*cdf0e10cSrcweir } 412*cdf0e10cSrcweir 413*cdf0e10cSrcweir void Model::removeBindingIfUseless( const XPropertySet_t& xBinding ) 414*cdf0e10cSrcweir throw( RuntimeException ) 415*cdf0e10cSrcweir { 416*cdf0e10cSrcweir Binding* pBinding = Binding::getBinding( xBinding ); 417*cdf0e10cSrcweir if( pBinding != NULL ) 418*cdf0e10cSrcweir { 419*cdf0e10cSrcweir if( ! pBinding->isUseful() ) 420*cdf0e10cSrcweir mpBindings->removeItem( pBinding ); 421*cdf0e10cSrcweir } 422*cdf0e10cSrcweir } 423*cdf0e10cSrcweir 424*cdf0e10cSrcweir Model::XDocument_t Model::newInstance( const rtl::OUString& sName, 425*cdf0e10cSrcweir const rtl::OUString& sURL, 426*cdf0e10cSrcweir sal_Bool bURLOnce ) 427*cdf0e10cSrcweir throw( RuntimeException ) 428*cdf0e10cSrcweir { 429*cdf0e10cSrcweir // create a default instance with <instanceData> element 430*cdf0e10cSrcweir XDocument_t xInstance = getDocumentBuilder()->newDocument(); 431*cdf0e10cSrcweir DBG_ASSERT( xInstance.is(), "failed to create DOM instance" ); 432*cdf0e10cSrcweir 433*cdf0e10cSrcweir Reference<XNode>( xInstance, UNO_QUERY_THROW )->appendChild( 434*cdf0e10cSrcweir Reference<XNode>( xInstance->createElement( OUSTRING("instanceData") ), 435*cdf0e10cSrcweir UNO_QUERY_THROW ) ); 436*cdf0e10cSrcweir 437*cdf0e10cSrcweir Sequence<PropertyValue> aSequence; 438*cdf0e10cSrcweir bool bOnce = bURLOnce; // bool, so we can take address in setInstanceData 439*cdf0e10cSrcweir setInstanceData( aSequence, &sName, &xInstance, &sURL, &bOnce ); 440*cdf0e10cSrcweir sal_Int32 nInstance = mpInstances->addItem( aSequence ); 441*cdf0e10cSrcweir loadInstance( nInstance ); 442*cdf0e10cSrcweir 443*cdf0e10cSrcweir return xInstance; 444*cdf0e10cSrcweir } 445*cdf0e10cSrcweir 446*cdf0e10cSrcweir sal_Int32 lcl_findProp( const PropertyValue* pValues, 447*cdf0e10cSrcweir sal_Int32 nLength, 448*cdf0e10cSrcweir const rtl::OUString& rName ) 449*cdf0e10cSrcweir { 450*cdf0e10cSrcweir bool bFound = false; 451*cdf0e10cSrcweir sal_Int32 n = 0; 452*cdf0e10cSrcweir for( ; !bFound && n < nLength; n++ ) 453*cdf0e10cSrcweir { 454*cdf0e10cSrcweir bFound = ( pValues[n].Name == rName ); 455*cdf0e10cSrcweir } 456*cdf0e10cSrcweir return bFound ? ( n - 1) : -1; 457*cdf0e10cSrcweir } 458*cdf0e10cSrcweir 459*cdf0e10cSrcweir sal_Int32 xforms::lcl_findInstance( const InstanceCollection* pInstances, 460*cdf0e10cSrcweir const rtl::OUString& rName ) 461*cdf0e10cSrcweir { 462*cdf0e10cSrcweir sal_Int32 nLength = pInstances->countItems(); 463*cdf0e10cSrcweir sal_Int32 n = 0; 464*cdf0e10cSrcweir bool bFound = false; 465*cdf0e10cSrcweir for( ; !bFound && n < nLength; n++ ) 466*cdf0e10cSrcweir { 467*cdf0e10cSrcweir OUString sName; 468*cdf0e10cSrcweir getInstanceData( pInstances->getItem( n ), &sName, NULL, NULL, NULL ); 469*cdf0e10cSrcweir bFound = ( sName == rName ); 470*cdf0e10cSrcweir } 471*cdf0e10cSrcweir return bFound ? ( n - 1 ) : -1; 472*cdf0e10cSrcweir } 473*cdf0e10cSrcweir 474*cdf0e10cSrcweir void Model::renameInstance( const rtl::OUString& sFrom, 475*cdf0e10cSrcweir const rtl::OUString& sTo, 476*cdf0e10cSrcweir const rtl::OUString& sURL, 477*cdf0e10cSrcweir sal_Bool bURLOnce ) 478*cdf0e10cSrcweir throw( RuntimeException ) 479*cdf0e10cSrcweir { 480*cdf0e10cSrcweir sal_Int32 nPos = lcl_findInstance( mpInstances, sFrom ); 481*cdf0e10cSrcweir if( nPos != -1 ) 482*cdf0e10cSrcweir { 483*cdf0e10cSrcweir Sequence<PropertyValue> aSeq = mpInstances->getItem( nPos ); 484*cdf0e10cSrcweir PropertyValue* pSeq = aSeq.getArray(); 485*cdf0e10cSrcweir sal_Int32 nLength = aSeq.getLength(); 486*cdf0e10cSrcweir 487*cdf0e10cSrcweir sal_Int32 nProp = lcl_findProp( pSeq, nLength, OUSTRING("ID") ); 488*cdf0e10cSrcweir if( nProp == -1 ) 489*cdf0e10cSrcweir { 490*cdf0e10cSrcweir // add name property 491*cdf0e10cSrcweir aSeq.realloc( nLength + 1 ); 492*cdf0e10cSrcweir pSeq = aSeq.getArray(); 493*cdf0e10cSrcweir pSeq[ nLength ].Name = OUSTRING("ID"); 494*cdf0e10cSrcweir nProp = nLength; 495*cdf0e10cSrcweir } 496*cdf0e10cSrcweir 497*cdf0e10cSrcweir // change name 498*cdf0e10cSrcweir pSeq[ nProp ].Value <<= sTo; 499*cdf0e10cSrcweir 500*cdf0e10cSrcweir // change url 501*cdf0e10cSrcweir nProp = lcl_findProp( pSeq, nLength, OUSTRING("URL") ); 502*cdf0e10cSrcweir if(nProp != -1) 503*cdf0e10cSrcweir pSeq[ nProp ].Value <<= sURL; 504*cdf0e10cSrcweir 505*cdf0e10cSrcweir // change urlonce 506*cdf0e10cSrcweir nProp = lcl_findProp( pSeq, nLength, OUSTRING("URLOnce") ); 507*cdf0e10cSrcweir if(nProp != -1) 508*cdf0e10cSrcweir pSeq[ nProp ].Value <<= bURLOnce; 509*cdf0e10cSrcweir 510*cdf0e10cSrcweir // set instance 511*cdf0e10cSrcweir mpInstances->setItem( nPos, aSeq ); 512*cdf0e10cSrcweir } 513*cdf0e10cSrcweir } 514*cdf0e10cSrcweir 515*cdf0e10cSrcweir void Model::removeInstance( const rtl::OUString& sName ) 516*cdf0e10cSrcweir throw( RuntimeException ) 517*cdf0e10cSrcweir { 518*cdf0e10cSrcweir sal_Int32 nPos = lcl_findInstance( mpInstances, sName ); 519*cdf0e10cSrcweir if( nPos != -1 ) 520*cdf0e10cSrcweir mpInstances->removeItem( mpInstances->getItem( nPos ) ); 521*cdf0e10cSrcweir } 522*cdf0e10cSrcweir 523*cdf0e10cSrcweir Reference<XNameContainer> lcl_getModels( 524*cdf0e10cSrcweir const Reference<com::sun::star::frame::XModel>& xComponent ) 525*cdf0e10cSrcweir { 526*cdf0e10cSrcweir Reference<XNameContainer> xRet; 527*cdf0e10cSrcweir Reference<XFormsSupplier> xSupplier( xComponent, UNO_QUERY ); 528*cdf0e10cSrcweir if( xSupplier.is() ) 529*cdf0e10cSrcweir { 530*cdf0e10cSrcweir xRet = xSupplier->getXForms(); 531*cdf0e10cSrcweir } 532*cdf0e10cSrcweir return xRet; 533*cdf0e10cSrcweir } 534*cdf0e10cSrcweir 535*cdf0e10cSrcweir Model::XModel_t Model::newModel( const Reference<com::sun::star::frame::XModel>& xCmp, 536*cdf0e10cSrcweir const OUString& sName ) 537*cdf0e10cSrcweir throw( RuntimeException ) 538*cdf0e10cSrcweir { 539*cdf0e10cSrcweir Model::XModel_t xModel; 540*cdf0e10cSrcweir Reference<XNameContainer> xModels = lcl_getModels( xCmp ); 541*cdf0e10cSrcweir if( xModels.is() 542*cdf0e10cSrcweir && ! xModels->hasByName( sName ) ) 543*cdf0e10cSrcweir { 544*cdf0e10cSrcweir Model* pModel = new Model(); 545*cdf0e10cSrcweir xModel.set( pModel ); 546*cdf0e10cSrcweir 547*cdf0e10cSrcweir pModel->setID( sName ); 548*cdf0e10cSrcweir pModel->newInstance( OUString(), OUString(), sal_False ); 549*cdf0e10cSrcweir pModel->initialize(); 550*cdf0e10cSrcweir xModels->insertByName( sName, makeAny( xModel ) ); 551*cdf0e10cSrcweir } 552*cdf0e10cSrcweir 553*cdf0e10cSrcweir return xModel; 554*cdf0e10cSrcweir } 555*cdf0e10cSrcweir 556*cdf0e10cSrcweir void Model::renameModel( const Reference<com::sun::star::frame::XModel>& xCmp, 557*cdf0e10cSrcweir const OUString& sFrom, 558*cdf0e10cSrcweir const OUString& sTo ) 559*cdf0e10cSrcweir throw( RuntimeException ) 560*cdf0e10cSrcweir { 561*cdf0e10cSrcweir Reference<XNameContainer> xModels = lcl_getModels( xCmp ); 562*cdf0e10cSrcweir if( xModels.is() 563*cdf0e10cSrcweir && xModels->hasByName( sFrom ) 564*cdf0e10cSrcweir && ! xModels->hasByName( sTo ) ) 565*cdf0e10cSrcweir { 566*cdf0e10cSrcweir Reference<XModel> xModel( xModels->getByName( sFrom ), UNO_QUERY ); 567*cdf0e10cSrcweir xModel->setID( sTo ); 568*cdf0e10cSrcweir xModels->insertByName( sTo, makeAny( xModel ) ); 569*cdf0e10cSrcweir xModels->removeByName( sFrom ); 570*cdf0e10cSrcweir } 571*cdf0e10cSrcweir } 572*cdf0e10cSrcweir 573*cdf0e10cSrcweir void Model::removeModel( const Reference<com::sun::star::frame::XModel>& xCmp, 574*cdf0e10cSrcweir const OUString& sName ) 575*cdf0e10cSrcweir throw( RuntimeException ) 576*cdf0e10cSrcweir { 577*cdf0e10cSrcweir Reference<XNameContainer> xModels = lcl_getModels( xCmp ); 578*cdf0e10cSrcweir if( xModels.is() 579*cdf0e10cSrcweir && xModels->hasByName( sName ) ) 580*cdf0e10cSrcweir { 581*cdf0e10cSrcweir xModels->removeByName( sName ); 582*cdf0e10cSrcweir } 583*cdf0e10cSrcweir } 584*cdf0e10cSrcweir 585*cdf0e10cSrcweir Model::XNode_t Model::createElement( const XNode_t& xParent, 586*cdf0e10cSrcweir const OUString& sName ) 587*cdf0e10cSrcweir throw( RuntimeException ) 588*cdf0e10cSrcweir { 589*cdf0e10cSrcweir Reference<XNode> xNode; 590*cdf0e10cSrcweir if( xParent.is() 591*cdf0e10cSrcweir && isValidXMLName( sName ) ) 592*cdf0e10cSrcweir { 593*cdf0e10cSrcweir // TODO: implement proper namespace handling 594*cdf0e10cSrcweir xNode.set( xParent->getOwnerDocument()->createElement( sName ), 595*cdf0e10cSrcweir UNO_QUERY ); 596*cdf0e10cSrcweir } 597*cdf0e10cSrcweir return xNode; 598*cdf0e10cSrcweir } 599*cdf0e10cSrcweir 600*cdf0e10cSrcweir Model::XNode_t Model::createAttribute( const XNode_t& xParent, 601*cdf0e10cSrcweir const OUString& sName ) 602*cdf0e10cSrcweir throw( RuntimeException ) 603*cdf0e10cSrcweir { 604*cdf0e10cSrcweir Reference<XNode> xNode; 605*cdf0e10cSrcweir Reference<XElement> xElement( xParent, UNO_QUERY ); 606*cdf0e10cSrcweir if( xParent.is() 607*cdf0e10cSrcweir && xElement.is() 608*cdf0e10cSrcweir && isValidXMLName( sName ) ) 609*cdf0e10cSrcweir { 610*cdf0e10cSrcweir // handle case where attribute already exists 611*cdf0e10cSrcweir sal_Int32 nCount = 0; 612*cdf0e10cSrcweir OUString sUniqueName = sName; 613*cdf0e10cSrcweir while( xElement->hasAttribute( sUniqueName ) ) 614*cdf0e10cSrcweir { 615*cdf0e10cSrcweir nCount++; 616*cdf0e10cSrcweir sUniqueName = sName + OUString::valueOf( nCount ); 617*cdf0e10cSrcweir } 618*cdf0e10cSrcweir 619*cdf0e10cSrcweir // TODO: implement proper namespace handling 620*cdf0e10cSrcweir xNode.set( xParent->getOwnerDocument()->createAttribute( sUniqueName ), 621*cdf0e10cSrcweir UNO_QUERY ); 622*cdf0e10cSrcweir } 623*cdf0e10cSrcweir return xNode; 624*cdf0e10cSrcweir } 625*cdf0e10cSrcweir 626*cdf0e10cSrcweir Model::XNode_t Model::renameNode( const XNode_t& xNode, 627*cdf0e10cSrcweir const rtl::OUString& sName ) 628*cdf0e10cSrcweir throw( RuntimeException ) 629*cdf0e10cSrcweir { 630*cdf0e10cSrcweir // early out if we don't have to change the name 631*cdf0e10cSrcweir if( xNode->getNodeName() == sName ) 632*cdf0e10cSrcweir return xNode; 633*cdf0e10cSrcweir 634*cdf0e10cSrcweir // refuse to change name if its an attribute, and the name is already used 635*cdf0e10cSrcweir if( xNode->getNodeType() == NodeType_ATTRIBUTE_NODE 636*cdf0e10cSrcweir && xNode->getParentNode().is() 637*cdf0e10cSrcweir && Reference<XElement>(xNode->getParentNode(), UNO_QUERY_THROW)->hasAttribute( sName ) ) 638*cdf0e10cSrcweir return xNode; 639*cdf0e10cSrcweir 640*cdf0e10cSrcweir // note old binding expression so we can adjust bindings below 641*cdf0e10cSrcweir OUString sOldDefaultBindingExpression = 642*cdf0e10cSrcweir getDefaultBindingExpressionForNode( xNode ); 643*cdf0e10cSrcweir 644*cdf0e10cSrcweir Reference<XDocument> xDoc = xNode->getOwnerDocument(); 645*cdf0e10cSrcweir Reference<XNode> xNew; 646*cdf0e10cSrcweir if( xNode->getNodeType() == NodeType_ELEMENT_NODE ) 647*cdf0e10cSrcweir { 648*cdf0e10cSrcweir Reference<XElement> xElem = xDoc->createElement( sName ); 649*cdf0e10cSrcweir xNew.set( xElem, UNO_QUERY ); 650*cdf0e10cSrcweir 651*cdf0e10cSrcweir // iterate over all attributes and append them to the new element 652*cdf0e10cSrcweir Reference<XElement> xOldElem( xNode, UNO_QUERY ); 653*cdf0e10cSrcweir OSL_ENSURE( xNode.is(), "no element?" ); 654*cdf0e10cSrcweir 655*cdf0e10cSrcweir Reference<XNamedNodeMap> xMap = xNode->getAttributes(); 656*cdf0e10cSrcweir sal_Int32 nLength = xMap.is() ? xMap->getLength() : 0; 657*cdf0e10cSrcweir for( sal_Int32 n = 0; n < nLength; n++ ) 658*cdf0e10cSrcweir { 659*cdf0e10cSrcweir Reference<XAttr> xAttr( xMap->item(n), UNO_QUERY ); 660*cdf0e10cSrcweir xElem->setAttributeNode( xOldElem->removeAttributeNode( xAttr ) ); 661*cdf0e10cSrcweir } 662*cdf0e10cSrcweir 663*cdf0e10cSrcweir // iterate over all children and append them to the new element 664*cdf0e10cSrcweir for( Reference<XNode> xCurrent = xNode->getFirstChild(); 665*cdf0e10cSrcweir xCurrent.is(); 666*cdf0e10cSrcweir xCurrent = xNode->getFirstChild() ) 667*cdf0e10cSrcweir { 668*cdf0e10cSrcweir xNew->appendChild( xNode->removeChild( xCurrent ) ); 669*cdf0e10cSrcweir } 670*cdf0e10cSrcweir 671*cdf0e10cSrcweir xNode->getParentNode()->replaceChild( xNew, xNode ); 672*cdf0e10cSrcweir } 673*cdf0e10cSrcweir else if( xNode->getNodeType() == NodeType_ATTRIBUTE_NODE ) 674*cdf0e10cSrcweir { 675*cdf0e10cSrcweir // create new attribute 676*cdf0e10cSrcweir Reference<XAttr> xAttr = xDoc->createAttribute( sName ); 677*cdf0e10cSrcweir xAttr->setValue( xNode->getNodeValue() ); 678*cdf0e10cSrcweir 679*cdf0e10cSrcweir // replace node 680*cdf0e10cSrcweir Reference<XNode> xParent = xNode->getParentNode(); 681*cdf0e10cSrcweir xParent->removeChild( xNode ); 682*cdf0e10cSrcweir xNew = xParent->appendChild( Reference<XNode>( xAttr, UNO_QUERY ) ); 683*cdf0e10cSrcweir } 684*cdf0e10cSrcweir else 685*cdf0e10cSrcweir { 686*cdf0e10cSrcweir OSL_ENSURE( false, "can't rename this node type" ); 687*cdf0e10cSrcweir } 688*cdf0e10cSrcweir 689*cdf0e10cSrcweir // adjust bindings (if necessary): 690*cdf0e10cSrcweir if( xNew.is() ) 691*cdf0e10cSrcweir { 692*cdf0e10cSrcweir // iterate over bindings and replace default expressions 693*cdf0e10cSrcweir OUString sNewDefaultBindingExpression = 694*cdf0e10cSrcweir getDefaultBindingExpressionForNode( xNew ); 695*cdf0e10cSrcweir for( sal_Int32 n = 0; n < mpBindings->countItems(); n++ ) 696*cdf0e10cSrcweir { 697*cdf0e10cSrcweir Binding* pBinding = Binding::getBinding( 698*cdf0e10cSrcweir mpBindings->Collection<XPropertySet_t>::getItem( n ) ); 699*cdf0e10cSrcweir 700*cdf0e10cSrcweir if( pBinding->getBindingExpression() 701*cdf0e10cSrcweir == sOldDefaultBindingExpression ) 702*cdf0e10cSrcweir pBinding->setBindingExpression( sNewDefaultBindingExpression ); 703*cdf0e10cSrcweir } 704*cdf0e10cSrcweir } 705*cdf0e10cSrcweir 706*cdf0e10cSrcweir // return node; return old node if renaming failed 707*cdf0e10cSrcweir return xNew.is() ? xNew : xNode; 708*cdf0e10cSrcweir } 709*cdf0e10cSrcweir 710*cdf0e10cSrcweir Model::XPropertySet_t Model::getBindingForNode( const XNode_t& xNode, 711*cdf0e10cSrcweir sal_Bool bCreate ) 712*cdf0e10cSrcweir throw( RuntimeException ) 713*cdf0e10cSrcweir { 714*cdf0e10cSrcweir OSL_ENSURE( xNode.is(), "no node?" ); 715*cdf0e10cSrcweir 716*cdf0e10cSrcweir // We will iterate over all bindings and determine the 717*cdf0e10cSrcweir // appropriateness of the respective binding for this node. The 718*cdf0e10cSrcweir // best one will be used. If we don't find any and bCreate is set, 719*cdf0e10cSrcweir // then we will create a suitable binding. 720*cdf0e10cSrcweir Binding* pBestBinding = NULL; 721*cdf0e10cSrcweir sal_Int32 nBestScore = 0; 722*cdf0e10cSrcweir 723*cdf0e10cSrcweir for( sal_Int32 n = 0; n < mpBindings->countItems(); n++ ) 724*cdf0e10cSrcweir { 725*cdf0e10cSrcweir Binding* pBinding = Binding::getBinding( 726*cdf0e10cSrcweir mpBindings->Collection<XPropertySet_t>::getItem( n ) ); 727*cdf0e10cSrcweir 728*cdf0e10cSrcweir OSL_ENSURE( pBinding != NULL, "no binding?" ); 729*cdf0e10cSrcweir Reference<XNodeList> xNodeList = pBinding->getXNodeList(); 730*cdf0e10cSrcweir 731*cdf0e10cSrcweir sal_Int32 nNodes = xNodeList.is() ? xNodeList->getLength() : 0; 732*cdf0e10cSrcweir if( nNodes > 0 && xNodeList->item( 0 ) == xNode ) 733*cdf0e10cSrcweir { 734*cdf0e10cSrcweir // allright, we found a suitable node. Let's determine how 735*cdf0e10cSrcweir // well it fits. Score: 736*cdf0e10cSrcweir // - bind to exactly this node is better than whole nodeset 737*cdf0e10cSrcweir // - simple binding expressions is better than complex ones 738*cdf0e10cSrcweir sal_Int32 nScore = 0; 739*cdf0e10cSrcweir if( nNodes == 1 ) 740*cdf0e10cSrcweir nScore ++; 741*cdf0e10cSrcweir if( pBinding->isSimpleBindingExpression() ) 742*cdf0e10cSrcweir nScore ++; 743*cdf0e10cSrcweir 744*cdf0e10cSrcweir // if we found a better binding, remember it 745*cdf0e10cSrcweir if( nScore > nBestScore ) 746*cdf0e10cSrcweir { 747*cdf0e10cSrcweir pBestBinding = pBinding; 748*cdf0e10cSrcweir nBestScore = nScore; 749*cdf0e10cSrcweir } 750*cdf0e10cSrcweir } 751*cdf0e10cSrcweir } 752*cdf0e10cSrcweir 753*cdf0e10cSrcweir // create binding, if none was found and bCreate is set 754*cdf0e10cSrcweir OSL_ENSURE( ( nBestScore == 0 ) == ( pBestBinding == NULL ), 755*cdf0e10cSrcweir "score != binding?" ); 756*cdf0e10cSrcweir if( bCreate && pBestBinding == NULL ) 757*cdf0e10cSrcweir { 758*cdf0e10cSrcweir pBestBinding = new Binding(); 759*cdf0e10cSrcweir pBestBinding->setBindingExpression( 760*cdf0e10cSrcweir getDefaultBindingExpressionForNode( xNode ) ); 761*cdf0e10cSrcweir mpBindings->addItem( pBestBinding ); 762*cdf0e10cSrcweir } 763*cdf0e10cSrcweir 764*cdf0e10cSrcweir return pBestBinding; 765*cdf0e10cSrcweir } 766*cdf0e10cSrcweir 767*cdf0e10cSrcweir void Model::removeBindingForNode( const XNode_t& ) 768*cdf0e10cSrcweir throw( RuntimeException ) 769*cdf0e10cSrcweir { 770*cdf0e10cSrcweir // determine whether suitable binding is still used 771*cdf0e10cSrcweir } 772*cdf0e10cSrcweir 773*cdf0e10cSrcweir OUString lcl_serializeForDisplay( const Reference< XAttr >& _rxAttrNode ) 774*cdf0e10cSrcweir { 775*cdf0e10cSrcweir ::rtl::OUString sResult; 776*cdf0e10cSrcweir OSL_ENSURE( _rxAttrNode.is(), "lcl_serializeForDisplay( attr ): invalid argument!" ); 777*cdf0e10cSrcweir if ( _rxAttrNode.is() ) 778*cdf0e10cSrcweir { 779*cdf0e10cSrcweir ::rtl::OUStringBuffer aBuffer; 780*cdf0e10cSrcweir aBuffer.append( _rxAttrNode->getName() ); 781*cdf0e10cSrcweir aBuffer.appendAscii( "=" ); 782*cdf0e10cSrcweir ::rtl::OUString sValue = _rxAttrNode->getValue(); 783*cdf0e10cSrcweir sal_Unicode nQuote = '"'; 784*cdf0e10cSrcweir if ( sValue.indexOf( nQuote ) >= 0 ) 785*cdf0e10cSrcweir nQuote = '\''; 786*cdf0e10cSrcweir aBuffer.append( nQuote ); 787*cdf0e10cSrcweir aBuffer.append( sValue ); 788*cdf0e10cSrcweir aBuffer.append( nQuote ); 789*cdf0e10cSrcweir aBuffer.append( (sal_Unicode)' ' ); 790*cdf0e10cSrcweir sResult = aBuffer.makeStringAndClear(); 791*cdf0e10cSrcweir } 792*cdf0e10cSrcweir return sResult; 793*cdf0e10cSrcweir } 794*cdf0e10cSrcweir 795*cdf0e10cSrcweir OUString lcl_serializeForDisplay( const Reference<XNodeList>& xNodes ) 796*cdf0e10cSrcweir { 797*cdf0e10cSrcweir ::rtl::OUString sResult; 798*cdf0e10cSrcweir 799*cdf0e10cSrcweir // create document fragment 800*cdf0e10cSrcweir Reference<XDocument> xDocument( getDocumentBuilder()->newDocument() ); 801*cdf0e10cSrcweir Reference<XDocumentFragment> xFragment( 802*cdf0e10cSrcweir xDocument->createDocumentFragment() ); 803*cdf0e10cSrcweir Reference<XNode> xNode( xFragment, UNO_QUERY ); 804*cdf0e10cSrcweir OSL_ENSURE( xFragment.is(), "xFragment" ); 805*cdf0e10cSrcweir OSL_ENSURE( xNode.is(), "xNode" ); 806*cdf0e10cSrcweir 807*cdf0e10cSrcweir sal_Int32 nAttributeNodes = 0; 808*cdf0e10cSrcweir 809*cdf0e10cSrcweir // attach nodelist to fragment 810*cdf0e10cSrcweir sal_Int32 nLength = xNodes->getLength(); 811*cdf0e10cSrcweir for( sal_Int32 i = 0; i < nLength; i++ ) 812*cdf0e10cSrcweir { 813*cdf0e10cSrcweir Reference<XNode> xCurrent = xNodes->item( i ); 814*cdf0e10cSrcweir 815*cdf0e10cSrcweir switch ( xCurrent->getNodeType() ) 816*cdf0e10cSrcweir { 817*cdf0e10cSrcweir case NodeType_DOCUMENT_NODE: 818*cdf0e10cSrcweir // special-case documents: use top-level element instead 819*cdf0e10cSrcweir xCurrent = xCurrent->getFirstChild(); 820*cdf0e10cSrcweir break; 821*cdf0e10cSrcweir case NodeType_ATTRIBUTE_NODE: 822*cdf0e10cSrcweir { 823*cdf0e10cSrcweir Reference< XAttr > xAttr( xCurrent, UNO_QUERY ); 824*cdf0e10cSrcweir if ( xAttr.is() ) 825*cdf0e10cSrcweir { 826*cdf0e10cSrcweir sResult += lcl_serializeForDisplay( xAttr ); 827*cdf0e10cSrcweir ++nAttributeNodes; 828*cdf0e10cSrcweir } 829*cdf0e10cSrcweir } 830*cdf0e10cSrcweir continue; 831*cdf0e10cSrcweir 832*cdf0e10cSrcweir default: 833*cdf0e10cSrcweir break; 834*cdf0e10cSrcweir } 835*cdf0e10cSrcweir 836*cdf0e10cSrcweir // append node 837*cdf0e10cSrcweir xNode->appendChild( xDocument->importNode( xCurrent, sal_True ) ); 838*cdf0e10cSrcweir } 839*cdf0e10cSrcweir OSL_ENSURE( ( nAttributeNodes == 0 ) || ( nAttributeNodes == nLength ), 840*cdf0e10cSrcweir "lcl_serializeForDisplay: mixed attribute and non-attribute nodes?" ); 841*cdf0e10cSrcweir if ( nAttributeNodes ) 842*cdf0e10cSrcweir // had only attribute nodes 843*cdf0e10cSrcweir return sResult; 844*cdf0e10cSrcweir 845*cdf0e10cSrcweir // serialize fragment 846*cdf0e10cSrcweir CSerializationAppXML aSerialization; 847*cdf0e10cSrcweir aSerialization.setSource( xFragment ); 848*cdf0e10cSrcweir aSerialization.serialize(); 849*cdf0e10cSrcweir 850*cdf0e10cSrcweir // copy stream into buffer 851*cdf0e10cSrcweir Reference<XTextInputStream> xTextInputStream( 852*cdf0e10cSrcweir createInstance( OUSTRING("com.sun.star.io.TextInputStream") ), 853*cdf0e10cSrcweir UNO_QUERY ); 854*cdf0e10cSrcweir Reference<XActiveDataSink>( xTextInputStream, UNO_QUERY_THROW ) 855*cdf0e10cSrcweir ->setInputStream( aSerialization.getInputStream() ); 856*cdf0e10cSrcweir 857*cdf0e10cSrcweir /* WORK AROUND for problem in serialization: currently, multiple 858*cdf0e10cSrcweir XML delarations (<?xml...?>) are being written out and we don't 859*cdf0e10cSrcweir want them. When this is fixed, the code below is nice and 860*cdf0e10cSrcweir simple. The current code filters out the declarations. 861*cdf0e10cSrcweir OUString sResult = xTextInputStream->readString( Sequence<sal_Unicode>(), 862*cdf0e10cSrcweir sal_True ); 863*cdf0e10cSrcweir */ 864*cdf0e10cSrcweir 865*cdf0e10cSrcweir // well, the serialization prepends XML header(s) that we need to 866*cdf0e10cSrcweir // remove first. 867*cdf0e10cSrcweir OUStringBuffer aBuffer; 868*cdf0e10cSrcweir while( ! xTextInputStream->isEOF() ) 869*cdf0e10cSrcweir { 870*cdf0e10cSrcweir OUString sLine = xTextInputStream->readLine(); 871*cdf0e10cSrcweir if( sLine.getLength() > 0 872*cdf0e10cSrcweir && sLine.compareToAscii( "<?xml", 5 ) != 0 ) 873*cdf0e10cSrcweir { 874*cdf0e10cSrcweir aBuffer.append( sLine ); 875*cdf0e10cSrcweir aBuffer.append( sal_Unicode('\n') ); 876*cdf0e10cSrcweir } 877*cdf0e10cSrcweir } 878*cdf0e10cSrcweir sResult = aBuffer.makeStringAndClear(); 879*cdf0e10cSrcweir 880*cdf0e10cSrcweir return sResult; 881*cdf0e10cSrcweir } 882*cdf0e10cSrcweir 883*cdf0e10cSrcweir OUString lcl_serializeForDisplay( const Reference<XXPathObject>& xResult ) 884*cdf0e10cSrcweir { 885*cdf0e10cSrcweir // error handling first 886*cdf0e10cSrcweir if( ! xResult.is() ) 887*cdf0e10cSrcweir return getResource( RID_STR_XFORMS_CANT_EVALUATE ); 888*cdf0e10cSrcweir 889*cdf0e10cSrcweir 890*cdf0e10cSrcweir // TODO: localize 891*cdf0e10cSrcweir OUStringBuffer aBuffer; 892*cdf0e10cSrcweir 893*cdf0e10cSrcweir switch( xResult->getObjectType() ) 894*cdf0e10cSrcweir { 895*cdf0e10cSrcweir case XPathObjectType_XPATH_BOOLEAN: 896*cdf0e10cSrcweir aBuffer.append( xResult->getBoolean() 897*cdf0e10cSrcweir ? OUSTRING("true") 898*cdf0e10cSrcweir : OUSTRING("false") ); 899*cdf0e10cSrcweir break; 900*cdf0e10cSrcweir 901*cdf0e10cSrcweir case XPathObjectType_XPATH_STRING: 902*cdf0e10cSrcweir aBuffer.append( sal_Unicode('"') ); 903*cdf0e10cSrcweir aBuffer.append( xResult->getString() ); 904*cdf0e10cSrcweir aBuffer.append( sal_Unicode('"') ); 905*cdf0e10cSrcweir break; 906*cdf0e10cSrcweir 907*cdf0e10cSrcweir case XPathObjectType_XPATH_NODESET: 908*cdf0e10cSrcweir aBuffer.append( lcl_serializeForDisplay( xResult->getNodeList() ) ); 909*cdf0e10cSrcweir break; 910*cdf0e10cSrcweir 911*cdf0e10cSrcweir case XPathObjectType_XPATH_NUMBER: 912*cdf0e10cSrcweir aBuffer.append( xResult->getDouble() ); 913*cdf0e10cSrcweir break; 914*cdf0e10cSrcweir 915*cdf0e10cSrcweir case XPathObjectType_XPATH_UNDEFINED: 916*cdf0e10cSrcweir case XPathObjectType_XPATH_POINT: 917*cdf0e10cSrcweir case XPathObjectType_XPATH_RANGE: 918*cdf0e10cSrcweir case XPathObjectType_XPATH_LOCATIONSET: 919*cdf0e10cSrcweir case XPathObjectType_XPATH_USERS: 920*cdf0e10cSrcweir case XPathObjectType_XPATH_XSLT_TREE: 921*cdf0e10cSrcweir default: 922*cdf0e10cSrcweir // TODO: localized error message? 923*cdf0e10cSrcweir break; 924*cdf0e10cSrcweir } 925*cdf0e10cSrcweir 926*cdf0e10cSrcweir return aBuffer.makeStringAndClear(); 927*cdf0e10cSrcweir } 928*cdf0e10cSrcweir 929*cdf0e10cSrcweir OUString Model::getResultForExpression( 930*cdf0e10cSrcweir const XPropertySet_t& xBinding, 931*cdf0e10cSrcweir sal_Bool bIsBindingExpression, 932*cdf0e10cSrcweir const OUString& sExpression ) 933*cdf0e10cSrcweir throw( RuntimeException ) 934*cdf0e10cSrcweir { 935*cdf0e10cSrcweir Binding* pBinding = Binding::getBinding( xBinding ); 936*cdf0e10cSrcweir if( pBinding == NULL ) 937*cdf0e10cSrcweir throw RuntimeException(); 938*cdf0e10cSrcweir 939*cdf0e10cSrcweir // prepare & evaluate expression 940*cdf0e10cSrcweir OUStringBuffer aBuffer; 941*cdf0e10cSrcweir ComputedExpression aExpression; 942*cdf0e10cSrcweir aExpression.setExpression( sExpression ); 943*cdf0e10cSrcweir if( bIsBindingExpression ) 944*cdf0e10cSrcweir { 945*cdf0e10cSrcweir // binding: use binding context and evaluation 946*cdf0e10cSrcweir aExpression.evaluate( pBinding->getEvaluationContext() ); 947*cdf0e10cSrcweir aBuffer.append( lcl_serializeForDisplay( aExpression.getXPath() ) ); 948*cdf0e10cSrcweir } 949*cdf0e10cSrcweir else 950*cdf0e10cSrcweir { 951*cdf0e10cSrcweir // MIP (not binding): iterate over bindings contexts 952*cdf0e10cSrcweir std::vector<EvaluationContext> aContext = 953*cdf0e10cSrcweir pBinding->getMIPEvaluationContexts(); 954*cdf0e10cSrcweir for( std::vector<EvaluationContext>::iterator aIter = aContext.begin(); 955*cdf0e10cSrcweir aIter != aContext.end(); 956*cdf0e10cSrcweir aIter ++ ) 957*cdf0e10cSrcweir { 958*cdf0e10cSrcweir aExpression.evaluate( *aIter ); 959*cdf0e10cSrcweir aBuffer.append( lcl_serializeForDisplay(aExpression.getXPath()) ); 960*cdf0e10cSrcweir aBuffer.append( sal_Unicode('\n') ); 961*cdf0e10cSrcweir } 962*cdf0e10cSrcweir } 963*cdf0e10cSrcweir return aBuffer.makeStringAndClear(); 964*cdf0e10cSrcweir } 965*cdf0e10cSrcweir 966*cdf0e10cSrcweir sal_Bool Model::isValidXMLName( const OUString& sName ) 967*cdf0e10cSrcweir throw( RuntimeException ) 968*cdf0e10cSrcweir { 969*cdf0e10cSrcweir return isValidQName( sName, NULL ); 970*cdf0e10cSrcweir } 971*cdf0e10cSrcweir 972*cdf0e10cSrcweir sal_Bool Model::isValidPrefixName( const OUString& sName ) 973*cdf0e10cSrcweir throw( RuntimeException ) 974*cdf0e10cSrcweir { 975*cdf0e10cSrcweir return ::isValidPrefixName( sName, NULL ); 976*cdf0e10cSrcweir } 977*cdf0e10cSrcweir 978*cdf0e10cSrcweir void Model::setNodeValue( 979*cdf0e10cSrcweir const XNode_t& xNode, 980*cdf0e10cSrcweir const rtl::OUString& sValue ) 981*cdf0e10cSrcweir throw( RuntimeException ) 982*cdf0e10cSrcweir { 983*cdf0e10cSrcweir setSimpleContent( xNode, sValue ); 984*cdf0e10cSrcweir } 985*cdf0e10cSrcweir 986*cdf0e10cSrcweir 987*cdf0e10cSrcweir // 988*cdf0e10cSrcweir // helper functions from model_helper.hxx 989*cdf0e10cSrcweir // 990*cdf0e10cSrcweir 991*cdf0e10cSrcweir void xforms::getInstanceData( 992*cdf0e10cSrcweir const Sequence<PropertyValue>& aValues, 993*cdf0e10cSrcweir OUString* pID, 994*cdf0e10cSrcweir Reference<XDocument>* pInstance, 995*cdf0e10cSrcweir OUString* pURL, 996*cdf0e10cSrcweir bool* pURLOnce ) 997*cdf0e10cSrcweir { 998*cdf0e10cSrcweir sal_Int32 nValues = aValues.getLength(); 999*cdf0e10cSrcweir const PropertyValue* pValues = aValues.getConstArray(); 1000*cdf0e10cSrcweir for( sal_Int32 n = 0; n < nValues; n++ ) 1001*cdf0e10cSrcweir { 1002*cdf0e10cSrcweir const PropertyValue& rValue = pValues[n]; 1003*cdf0e10cSrcweir #define PROP(NAME) \ 1004*cdf0e10cSrcweir if( p##NAME != NULL && \ 1005*cdf0e10cSrcweir rValue.Name.equalsAsciiL(RTL_CONSTASCII_STRINGPARAM(#NAME)) ) \ 1006*cdf0e10cSrcweir rValue.Value >>= (*p##NAME) 1007*cdf0e10cSrcweir PROP(ID); 1008*cdf0e10cSrcweir PROP(Instance); 1009*cdf0e10cSrcweir PROP(URL); 1010*cdf0e10cSrcweir PROP(URLOnce); 1011*cdf0e10cSrcweir #undef PROP 1012*cdf0e10cSrcweir } 1013*cdf0e10cSrcweir } 1014*cdf0e10cSrcweir 1015*cdf0e10cSrcweir void xforms::setInstanceData( 1016*cdf0e10cSrcweir Sequence<PropertyValue>& aSequence, 1017*cdf0e10cSrcweir const OUString* _pID, 1018*cdf0e10cSrcweir const Reference<XDocument>* _pInstance, 1019*cdf0e10cSrcweir const OUString* _pURL, 1020*cdf0e10cSrcweir const bool* _pURLOnce ) 1021*cdf0e10cSrcweir { 1022*cdf0e10cSrcweir // get old instance data 1023*cdf0e10cSrcweir OUString sID; 1024*cdf0e10cSrcweir Reference<XDocument> xInstance; 1025*cdf0e10cSrcweir OUString sURL; 1026*cdf0e10cSrcweir bool bURLOnce = false; 1027*cdf0e10cSrcweir getInstanceData( aSequence, &sID, &xInstance, &sURL, &bURLOnce ); 1028*cdf0e10cSrcweir const OUString* pID = ( sID.getLength() > 0 ) ? &sID : NULL; 1029*cdf0e10cSrcweir const Reference<XDocument>* pInstance = xInstance.is() ? &xInstance : NULL; 1030*cdf0e10cSrcweir const OUString* pURL = ( sURL.getLength() > 0 ) ? &sURL : NULL; 1031*cdf0e10cSrcweir const bool* pURLOnce = ( bURLOnce && pURL != NULL ) ? &bURLOnce : NULL; 1032*cdf0e10cSrcweir 1033*cdf0e10cSrcweir // determine new instance data 1034*cdf0e10cSrcweir #define PROP(NAME) if( _p##NAME != NULL ) p##NAME = _p##NAME 1035*cdf0e10cSrcweir PROP(ID); 1036*cdf0e10cSrcweir PROP(Instance); 1037*cdf0e10cSrcweir PROP(URL); 1038*cdf0e10cSrcweir PROP(URLOnce); 1039*cdf0e10cSrcweir #undef PROP 1040*cdf0e10cSrcweir 1041*cdf0e10cSrcweir // count # of values we want to set 1042*cdf0e10cSrcweir sal_Int32 nCount = 0; 1043*cdf0e10cSrcweir #define PROP(NAME) if( p##NAME != NULL ) nCount++ 1044*cdf0e10cSrcweir PROP(ID); 1045*cdf0e10cSrcweir PROP(Instance); 1046*cdf0e10cSrcweir PROP(URL); 1047*cdf0e10cSrcweir PROP(URLOnce); 1048*cdf0e10cSrcweir #undef PROP 1049*cdf0e10cSrcweir 1050*cdf0e10cSrcweir // realloc sequence and enter values; 1051*cdf0e10cSrcweir aSequence.realloc( nCount ); 1052*cdf0e10cSrcweir PropertyValue* pSequence = aSequence.getArray(); 1053*cdf0e10cSrcweir sal_Int32 nIndex = 0; 1054*cdf0e10cSrcweir #define PROP(NAME) \ 1055*cdf0e10cSrcweir if( p##NAME != NULL ) \ 1056*cdf0e10cSrcweir { \ 1057*cdf0e10cSrcweir pSequence[ nIndex ].Name = OUSTRING(#NAME); \ 1058*cdf0e10cSrcweir pSequence[ nIndex ].Value <<= *p##NAME; \ 1059*cdf0e10cSrcweir nIndex++; \ 1060*cdf0e10cSrcweir } 1061*cdf0e10cSrcweir PROP(ID); 1062*cdf0e10cSrcweir PROP(Instance); 1063*cdf0e10cSrcweir PROP(URL); 1064*cdf0e10cSrcweir PROP(URLOnce); 1065*cdf0e10cSrcweir #undef PROP 1066*cdf0e10cSrcweir } 1067