webservices/wsxml/src/senbaseelement.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Thu, 15 Jul 2010 19:53:58 +0300
branchRCL_3
changeset 34 f68f07157250
parent 0 62f9d29f7211
child 36 c5fabff9b552
permissions -rw-r--r--
Revision: 201025 Kit: 2010127

/*
* Copyright (c) 2002-2005 Nokia Corporation and/or its subsidiary(-ies). 
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of "Eclipse Public License v1.0"
* which accompanies this distribution, and is available
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
*
* Initial Contributors:
* Nokia Corporation - initial contribution.
*
* Contributors:
*
* Description:        
*
*/








// INCLUDE FILES
#include "SenBaseElement.h"
#include "SenXmlUtils.h"
#include "senxmldebug.h"

#include "SenXmlConstants.h" // KSenColon, ++



#ifdef SYMBIAN_SECURE_ECOM 
    // for S60 2nd edition FP3 (2.8) OR newer 3rd edition platform
    #include <xml/attribute.h> // needed for RAttributeArray
#else 
    // for S60 2nd edition, up to FP2 (2.0 - 2.6)
    #include "Attribute.h" // needed for RAttributeArray
#endif

using namespace Xml;

namespace
    {
/*
    _LIT8(KColon,            ":");
    _LIT8(KLessThan,         "<");
    _LIT8(KGreaterThan,      ">");
    _LIT8(KSpace,            " ");
    _LIT8(KDblQuot,          "\"");
    _LIT8(KEqualsDblQuot,    "=\"");
    _LIT8(KSlashGreaterThan, "/>");
    _LIT8(KLessThanSlash,    "</");
    _LIT8(KSpaceXmlns,       " xmlns");
    _LIT8(KXmlns,            "xmlns");
    _LIT8(KXmlNsAttNamePlusColon,   "xmlns:"); 
*/
    const TInt KFlatBufSize          = 64;  // CBufFlat
    }

EXPORT_C CSenBaseElement* CSenBaseElement::NewL(const TDesC8& aLocalName)
    {
    CSenBaseElement* pNew = new (ELeave) CSenBaseElement;
    CleanupStack::PushL(pNew);
    pNew->BaseConstructL(aLocalName);
    CleanupStack::Pop(); // pNew;
    return pNew;
    }

EXPORT_C CSenBaseElement* CSenBaseElement::NewL(const TDesC8& aNsUri,
                                                const TDesC8& aLocalName)
    {
    CSenBaseElement* pNew = new (ELeave) CSenBaseElement;
    CleanupStack::PushL(pNew);
    pNew->BaseConstructL(aNsUri, aLocalName);
    CleanupStack::Pop(); // pNew;
    return pNew;
    }

EXPORT_C CSenBaseElement* CSenBaseElement::NewL(const TDesC8& aNsUri,
                                                const TDesC8& aLocalName,
                                                const TDesC8& aQName)
    {
    CSenBaseElement* pNew = new (ELeave) CSenBaseElement;
    CleanupStack::PushL(pNew);
    pNew->BaseConstructL(aNsUri, aLocalName, aQName);
    CleanupStack::Pop(); // pNew;
    return pNew;
    }

EXPORT_C CSenBaseElement* CSenBaseElement::NewL(const TDesC8& aNsUri,
                                                const TDesC8& aLocalName,
                                                const TDesC8& aQName,
                                                const RAttributeArray& apAttrs)
    {
    CSenBaseElement* pNew = new (ELeave) CSenBaseElement;
    CleanupStack::PushL(pNew);
    pNew->BaseConstructL(aNsUri, aLocalName, aQName, apAttrs);
    CleanupStack::Pop(); // pNew;
    return pNew;
    }


EXPORT_C CSenBaseElement* CSenBaseElement::NewL(const TDesC8& aNsUri,
                                                const TDesC8& aLocalName,
                                                const TDesC8& aQName,
                                                const RAttributeArray& apAttrs,
                                                CSenElement& aParent)
    {
    CSenBaseElement* pNew = new (ELeave) CSenBaseElement;
    CleanupStack::PushL(pNew);
    pNew->BaseConstructL(aNsUri, aLocalName, aQName, apAttrs, aParent);
    CleanupStack::Pop(); // pNew;
    return pNew;
    }

EXPORT_C CSenBaseElement::CSenBaseElement()
: ipLocalName(NULL),
//  ipNsUri(NULL),
  ipContentBuf(NULL),
  ipParent(NULL),
  ipNamespace(NULL)
    {
    }

EXPORT_C CSenBaseElement::~CSenBaseElement()
    {
    delete ipLocalName;
//    delete ipNsUri;
    delete ipContentBuf;

    iAttrs.ResetAndDestroy();

    iElements.ResetAndDestroy();
    iNamespaces.ResetAndDestroy();
    }



EXPORT_C void CSenBaseElement::BaseConstructL(const TDesC8& aLocalName)
    {
    SenXmlUtils::LeaveOnInvalidElementNameL(aLocalName);
    ipLocalName = aLocalName.AllocL();
    }

EXPORT_C void CSenBaseElement::BaseConstructL(const TDesC8& aNsUri,
                                              const TDesC8& aLocalName)
    {
    SenXmlUtils::LeaveOnInvalidElementNameL(aLocalName);
    if(aNsUri.Length()>0)
        {
        SetNamespaceL(aNsUri);
        }
    ipLocalName = aLocalName.AllocL();
    }

EXPORT_C void CSenBaseElement::BaseConstructL(  const TDesC8& aNsUri,
                                                const TDesC8& aLocalName,
                                                const TDesC8& aQName)
    {
    SenXmlUtils::LeaveOnInvalidElementNameL(aLocalName);
    SenXmlUtils::LeaveOnInvalidElementNameL(aQName);
    ipLocalName = aLocalName.AllocL();
    TPtrC8 prefix(KNullDesC8);
    if(aQName.Length()>0)
        {
        TInt colon = aQName.Locate(':');
        if(colon>0) // Note: 0 also treated as no prefix
            {
            prefix.Set(aQName.Ptr(), colon);
            }
        }
    SetNamespaceL(prefix, aNsUri);
    }

EXPORT_C void CSenBaseElement::BaseConstructL(  const TDesC8& aNsUri,
                                                const TDesC8& aLocalName,
                                                const TDesC8& aQName,
                                                const RAttributeArray& apAttrs)
    {
    BaseConstructL(aNsUri, aLocalName, aQName);
    SetAttributesL(apAttrs);
    }

EXPORT_C void CSenBaseElement::BaseConstructL(  const TDesC8& aNsUri,
                                                const TDesC8& aLocalName,
                                                const TDesC8& aQName,
                                                const RAttributeArray& apAttrs,
                                                CSenElement& aParent    )
    {
    // parent must be set here at first line, because
    // namespace setting dependends of it(!)
    ipParent = &aParent;

    BaseConstructL(aNsUri, aLocalName, aQName);
    SetAttributesL(apAttrs);
    }

EXPORT_C void CSenBaseElement::SetAttributesL(const RAttributeArray& apAttrs)
    {
    AddAttributesL(apAttrs);
    }


EXPORT_C void CSenBaseElement::AddAttributesL(const RAttributeArray& apAttrs)
    {
    TInt count(apAttrs.Count());
    HBufC8* qName = NULL;
    for(TInt i=0; i<count; i++)
        {
        const TPtrC8 localname = apAttrs[i].Attribute().LocalName().DesC();
        const TPtrC8 prefix = apAttrs[i].Attribute().Prefix().DesC();
        const TPtrC8 value = apAttrs[i].Value().DesC();

        SenXmlUtils::BuildQNameL(prefix, localname, qName);

        CleanupStack::PushL(qName);

        HBufC8* encodedValue = NULL;
        TBool encoded = 
            SenXmlUtils::EncodeHttpCharactersL(value, encodedValue);

        if (encoded)
            {
            CleanupStack::PushL(encodedValue);
            AddAttributeL(*qName, localname, *encodedValue); 
            // note: this intentionally
            // skips the 2 param 
            // AddAttributeL() function

            CleanupStack::PopAndDestroy(); // encodedValue
            }
        else
            {
            AddAttributeL(*qName, localname, value); 
            // note: this intentionally
            // skips the 2 param 
            // AddAttributeL() function
            }

        CleanupStack::PopAndDestroy(); // qName
        }
    }

EXPORT_C const TDesC8& CSenBaseElement::AddAttributeL(const TDesC8& aQName,
                                                     const TDesC8& aLocalName,
                                                     const TDesC8& aValue)
    {
    // first we check for possible namespace declarations...
    if ( aQName == KSenXmlns || aLocalName == KSenXmlns)
        {
        // this is a default name space declaration
        SetNamespaceL(aValue);
        }
    else if (aLocalName == NsPrefix())
        {
        //we have a new declaration for the namespace of this element
        SetNamespaceL(aLocalName, aValue);
        }
    else if(aQName.Find(KSenXmlNsAttNamePlusColon) == 0)
        {
        // we have an additional namespace declaration
        AddNamespaceL(aLocalName, aValue);
        }
    else
        {
        // we have a real attribute!
        CSenBaseAttribute* pAttribute = CSenBaseAttribute::NewL(aQName,
                                                                aLocalName,
                                                                aValue);
        CleanupStack::PushL(pAttribute);
        AddAttributeL(pAttribute);
        CleanupStack::Pop(pAttribute);
        }
    return aValue;
    }



EXPORT_C const TDesC8& CSenBaseElement::AddAttributeL(const TDesC8& aAttrName,
                                                      const TDesC8& aValue)
    {
    // note, the aAttrName may be a qualified name or simply localname
    // strip off the possible prefix from possible qualified name:
    TPtrC8 localName = SenXmlUtils::LocalName(aAttrName);
    return AddAttributeL(aAttrName, localName, aValue);
    }

// takes the ownership of aAttribute
EXPORT_C const TDesC8& CSenBaseElement::AddAttributeL(CSenBaseAttribute* apAttribute)
    {
    CSenBaseAttribute* pOldAtt = FindAttr(apAttribute->Name());

    if(!pOldAtt)
        {
        // transfrer the ownership to this class:
#ifdef EKA2
        iAttrs.AppendL(apAttribute);
#else
        User::LeaveIfError(iAttrs.Append(apAttribute));
#endif // EKA2
        return apAttribute->Value();
        }
    else
        {
        pOldAtt->SetValueL(apAttribute->Value());
        delete apAttribute;
        apAttribute = NULL;
        return pOldAtt->Value();
        }
    }


EXPORT_C CSenBaseAttribute* CSenBaseElement::FindAttr(const TDesC8& aName)
    {
    for (TInt i = 0; i < iAttrs.Count(); i++)
        {
        if (iAttrs[i]->Name() == aName)
            {
            return iAttrs[i];
            }
        }
    return NULL;
    }


EXPORT_C TInt CSenBaseElement::IndexOfElement(const TDesC8& aNsUri,
                                              const TDesC8& aLocalName) const
    {
    for (TInt i = 0; i < iElements.Count(); i++)
        {
        CSenElement* pElement = iElements[i];
        const TDesC8& nsUri = pElement->NamespaceURI();
        const TDesC8& localName = pElement->LocalName();
        if ((nsUri == aNsUri) && (localName == aLocalName))
            {
            return i;
            }
        }
    return KErrNotFound;
    }

EXPORT_C const TDesC8& CSenBaseElement::LocalName() const
    {
    if (ipLocalName == NULL)
        {
        return KNullDesC8();
        }
    else
        {
        return *ipLocalName;
        }
    }

EXPORT_C const TDesC8& CSenBaseElement::NamespaceURI() const
    {
    if (ipNamespace)
        {
        return ipNamespace->URI();
        }
    else
        {
        return KNullDesC8();
        }
    }

EXPORT_C const TDesC8& CSenBaseElement::NsPrefix() const
    {
    if (ipNamespace == NULL)
        {
        return KNullDesC8();
        }
    else
        {
        return ipNamespace->Prefix();
        }
    }

EXPORT_C TBool CSenBaseElement::HasContent() const
    {
    if (ipContentBuf == NULL)
        {
        return EFalse;
        }
    else
        {
        return (ipContentBuf->Size() > 0);
        }
    }

EXPORT_C TPtrC8 CSenBaseElement::Content() const
    {
    if (ipContentBuf == NULL)
        {
        return KNullDesC8();
        }
    else
        {
        TPtrC8 p8 = ipContentBuf->Ptr(0);
        return p8;
        }
    }


EXPORT_C HBufC* CSenBaseElement::ContentUnicodeL() const
    {
    HBufC* pRet = SenXmlUtils::ToUnicodeLC(Content());
    CleanupStack::Pop(); // pRet;
    return pRet;
    }

EXPORT_C TPtrC8 CSenBaseElement::SetContentL(const TDesC8& aContent)
    {
    AllocContentBufL(); // content length is _OK_ to be ZERO
    ipContentBuf->ResizeL(aContent.Size());
    ipContentBuf->Write(0, aContent);
  
    //   TPtrC8(REINTERPRET_CAST(const TUint8*, aContent.Ptr()),
    //                                            aContent.Size()));
    // Reset stream
    iContentWriteStream.Open(*ipContentBuf);
    return Content();
    }

EXPORT_C RWriteStream& CSenBaseElement::ContentWriteStreamL()
    {
    AllocContentBufL();
    iContentWriteStream.Open(*ipContentBuf);
    return iContentWriteStream;
    }

// NOTE: assumes(!) that the namespace is same(!)
EXPORT_C RPointerArray<CSenElement>& CSenBaseElement::ElementsL()
    {
    return iElements;
    }

EXPORT_C TInt CSenBaseElement::ElementsL(
                                    RPointerArray<CSenElement>& aElementArray,
                                    const TDesC8& aNsUri,
                                    const TDesC8& aLocalName)
    {
    if (iElements.Count() == 0)
        return KErrNotFound;

    CSenElement* pElement = NULL;
    TInt count(iElements.Count());
    for (TInt i=0; i<count; i++)
        {
        pElement = iElements[i];
        if ( (aLocalName == pElement->LocalName()) &&
             (aNsUri == pElement->NamespaceURI()) )
            {
#ifdef EKA2
            aElementArray.AppendL(pElement);
#else
            User::LeaveIfError(aElementArray.Append(pElement));
#endif // EKA2
            }
        }

    return KErrNone;
    }

EXPORT_C TInt CSenBaseElement::ElementsL(
                                    RPointerArray<CSenElement>& aElementArray,
                                    const TDesC8& aLocalName)
    {
    return ElementsL(aElementArray,NamespaceURI(),aLocalName);
    }

EXPORT_C const TDesC8* CSenBaseElement::AttrValue(const TDesC8& aName)
    {
    CSenBaseAttribute* pAttr = FindAttr(aName);
    if (pAttr == NULL)
        {
        return NULL;
        }
    else
        {
        return &(pAttr->Value());
        }
    }

EXPORT_C void CSenBaseElement::AddAttrL(const TDesC8& aName,
                                        const TDesC8& aValue)
    {
    CSenBaseAttribute* pAttr = FindAttr(aName);
    if (pAttr == NULL)
        {
        //Create new baseAttribute object
        CSenBaseAttribute* newBaseAttribute = CSenBaseAttribute::NewL(aName, aValue);
        //Push to cleanupstack 
        CleanupStack::PushL(newBaseAttribute);
        
        //Add the new baseAttribute to existing Attribute list
        User::LeaveIfError(iAttrs.Append(newBaseAttribute));
        
        //Pop from CleanupStack
        CleanupStack::Pop(newBaseAttribute);
        }
    else
        {
        pAttr->SetValueL(aValue);
        }
    }

EXPORT_C CSenElement* CSenBaseElement::Parent()
    {
    return ipParent;
    }

EXPORT_C CSenElement* CSenBaseElement::SetParent(CSenElement* apParent)  // IOP
    {
    if (apParent && ipParent != apParent)
        {
        ipParent = apParent;
        if (!ipNamespace)
            {
            // check if there is a default namespace declared in the scope of
            // the parent
            const CSenNamespace* pParentNamespace =
                ((CSenBaseElement*) ipParent)->Namespace(KNullDesC8,ETrue);
            if (pParentNamespace && pParentNamespace->Prefix() == KNullDesC8)
                {
                ipNamespace = (CSenNamespace*)pParentNamespace;
                }
            }
        else
            {
            //check if the parent already has a namespace for this element
            // if so remove it from the local namespace table
            const CSenNamespace* pNs = ipParent->Namespace(KNullDesC8,
                                                    ipNamespace->URI());
            if (pNs && pNs != ipNamespace)
                {
                if (ipNamespace->Compare(*pNs) ||
                    ipNamespace->Prefix().Length() == 0)
                    {
                    //prefix is also identical or this element has no prefix
                    TInt nsIndex = iNamespaces.Find(ipNamespace);
                    if ( nsIndex != KErrNotFound)
                        {
                        iNamespaces.Remove(nsIndex);
                        }
                    delete ipNamespace;
                    ipNamespace = (CSenNamespace*)pNs;
                    }
                }

            }
        }
    return apParent;
    }

EXPORT_C MSenElement& CSenBaseElement::Root()
    {
    if (ipParent == NULL)
        {
        return *this;
        }
    else
        {
        return ipParent->Root();
        }
    }

EXPORT_C CSenElement* CSenBaseElement::Element(const TDesC8& aLocalName)
    {
    return Element(NamespaceURI(), aLocalName);
    }

EXPORT_C CSenElement* CSenBaseElement::Element(const TDesC8& aNsUri,
                                               const TDesC8& aLocalName)
    {
    TInt idx = IndexOfElement(aNsUri, aLocalName);
    if (idx < 0)
        {
        return NULL;
        }
    else
        {
        return iElements[idx];
        }
    }

EXPORT_C CSenElement* CSenBaseElement::CreateElementL(const TDesC8& aNsPrefix,
                                                      const TDesC8& aLocalName)
    {
    CSenElement* pNewElement = NULL;

    if (aNsPrefix.Length() > 0)
        {
        CSenNamespace* pNamespace = (CSenNamespace*)Namespace(aNsPrefix);
        if (pNamespace)
            {
            HBufC8 *pQName =
                HBufC8::NewLC(aNsPrefix.Length() + aLocalName.Length() +5);
            TPtr8 ptr = pQName->Des();
            ptr.Append(aNsPrefix);
            ptr.Append(':');
            ptr.Append(aLocalName);
            pNewElement =
                CSenBaseElement::NewL(pNamespace->URI(), aLocalName, *pQName);
            CleanupStack::PopAndDestroy(); // pQName
            }
        }
    else
        {
        pNewElement = CSenBaseElement::NewL(aLocalName);
        }

    return pNewElement; // Returns NULL if required namespace can not be found!
    }

EXPORT_C CSenElement& CSenBaseElement::InsertElementL(
                                          CSenElement& aElement,
                                          const CSenElement& aBeforeElement)
    {
    TInt index(KErrNotFound);
    index = iElements.Find(&aBeforeElement);
    if (index != KErrNotFound)
        {
        User::LeaveIfError(iElements.Insert(&aElement,index));
        }
    else
        {
        User::LeaveIfError(iElements.Append(&aElement));
        }
    aElement.SetParent(this);
    return aElement;
    }

EXPORT_C CSenElement& CSenBaseElement::AddElementL(CSenElement& aElement)
    {
    User::LeaveIfError(iElements.Append(&aElement));
    aElement.SetParent(this);
    return aElement;
    }

EXPORT_C CSenElement& CSenBaseElement::AddElementL(const TDesC8& aNsUri,
                                                   const TDesC8& aLocalName)
    {
    //Create new baseElement
    CSenBaseElement *newBaseElement = CSenBaseElement::NewL(aNsUri, aLocalName);
    //Push it to CleanupStack
    CleanupStack::PushL(newBaseElement);
    
    //Add the baseElement
    CSenElement& newAddedElement  = AddElementL(*newBaseElement);
    
    //Pop baseElement from CleanupStack
    CleanupStack::Pop(newBaseElement);
    
    //Return the newly added element (got from the AddElement() API)
    return newAddedElement;
    }

EXPORT_C CSenElement& CSenBaseElement::AddElementL(
    const TDesC8& aNsUri,
    const TDesC8& aLocalName,
    const TDesC8& aQName
    )
    {
    //Create new baseElement
    CSenBaseElement *newBaseElement = CSenBaseElement::NewL(aNsUri, aLocalName, aQName);
    //Push it to CleanupStack
    CleanupStack::PushL(newBaseElement);

    //Add the baseElement
    CSenElement& newAddedElement  =  AddElementL(*newBaseElement);
    
    //Pop baseElement from CleanupStack
    CleanupStack::Pop(newBaseElement);
        
    //Return the newly added element (got from the AddElement() API)
    return newAddedElement;
    }

EXPORT_C CSenElement& CSenBaseElement::AddElementL(const TDesC8& aLocalName)
    {
    //Create new baseElement
    CSenBaseElement *newBaseElement = CSenBaseElement::NewL(aLocalName);
    //Push it to CleanupStack
    CleanupStack::PushL(newBaseElement);
    
    //Add the baseElement
    CSenElement& newAddedElement  =  AddElementL(*newBaseElement);
    
    //Pop baseElement from CleanupStack
    CleanupStack::Pop(newBaseElement);
       
    //Return the newly added element (got from the AddElement() API)
    return newAddedElement;
    }

EXPORT_C CSenElement* CSenBaseElement::RemoveElement(CSenElement& aElement)
    {
    TInt idx = iElements.Find(&aElement);
    if (idx >= 0)
        {
        CSenElement* pElement = iElements[idx];
        iElements.Remove(idx);
        pElement->SetParent(NULL);
        return pElement;
        }
    else
        {
        return NULL;
        }
    }

EXPORT_C CSenElement* CSenBaseElement::RemoveElement(const TDesC8& aNsUri,
                                                     const TDesC8& aLocalName)
    {
    TInt idx = IndexOfElement(aNsUri, aLocalName);
    if (idx >= 0)
        {
        CSenElement* pElement = iElements[idx];
        iElements.Remove(idx);
        pElement->SetParent(NULL);
        return pElement;
        }
    else
        {
        return NULL;
        }
    }

EXPORT_C CSenElement* CSenBaseElement::RemoveElement(const TDesC8& aLocalName)
    {
    return RemoveElement(NamespaceURI(), aLocalName);
    }

EXPORT_C CSenElement* CSenBaseElement::ReplaceElementL(CSenElement& aElement)
    {
    CSenElement* pOldElement =
                RemoveElement(aElement.NamespaceURI(), aElement.LocalName());
    CleanupStack::PushL(pOldElement);
    AddElementL(aElement);
    CleanupStack::Pop(); // pOldElement;
    return pOldElement;
    }

EXPORT_C HBufC8* CSenBaseElement::AsXmlL()
    {
    CBufFlat *pBuf = CBufFlat::NewL(KFlatBufSize);
    CleanupStack::PushL(pBuf);
    TPtrC8 p = WriteToBufL(*pBuf);
    HBufC8* pRet = p.AllocL();
    CleanupStack::PopAndDestroy(); // pBuf;
    return pRet;

    }

EXPORT_C HBufC* CSenBaseElement::AsXmlUnicodeL()
    {
    CBufFlat *pBuf = CBufFlat::NewL(KFlatBufSize);
    CleanupStack::PushL(pBuf);
    TPtrC8 p8 = WriteToBufL(*pBuf);
    HBufC* pRet = SenXmlUtils::ToUnicodeLC(p8);
    CleanupStack::Pop(); // pop pRet;
    CleanupStack::PopAndDestroy(); // pBuf;
    return pRet;
    }

EXPORT_C TPtrC8 CSenBaseElement::WriteToBufL(CBufBase& aBuf)
    {
    RBufWriteStream bufWs(aBuf);
    CleanupClosePushL(bufWs);
    this->WriteAsXMLToL(bufWs);
    CleanupStack::PopAndDestroy(); // bufWs.Close();
    return aBuf.Ptr(0);
    }

EXPORT_C void CSenBaseElement::WriteAsXMLToL(RWriteStream& aWriteStream)
    {
    // Find out whether we should declare the namespace
    TPtrC8 nsPrefix = NsPrefix();

    // Element name
    aWriteStream.WriteL(KSenLessThan);
    if (nsPrefix.Length() > 0)
        {
        aWriteStream.WriteL(nsPrefix);
        aWriteStream.WriteL(KSenColon);
        }
    aWriteStream.WriteL(*ipLocalName);

    if (iAttrs.Count() > 0 || iNamespaces.Count() > 0)
        {
        WriteNamespacesToL(aWriteStream);
        WriteAttrsToL(aWriteStream);
        }

    // Elements and content
    if ((iElements.Count() > 0) || HasContent())
        {
        aWriteStream.WriteL(KSenGreaterThan);

        // Body
        WriteElementsToL(aWriteStream);
        WriteContentToL(aWriteStream);

        // Closing element
        aWriteStream.WriteL(KSenLessThanSlash);
        if (nsPrefix.Length() > 0)
            {
            aWriteStream.WriteL(nsPrefix);
            aWriteStream.WriteL(KSenColon);
            }
        aWriteStream.WriteL(*ipLocalName);
        aWriteStream.WriteL(KSenGreaterThan);
        }
    else
        {
        aWriteStream.WriteL(KSenSlashGreaterThan);
        }
    }

EXPORT_C void CSenBaseElement::WriteAttrToL(RWriteStream& aWriteStream,
                                            const TDesC8& aName,
                                            const TDesC8& aValue)
    {
    aWriteStream.WriteL(KSenSpace);
    aWriteStream.WriteL(aName);
    aWriteStream.WriteL(KSenEqualsDblQuot);
    aWriteStream.WriteL(aValue);
    aWriteStream.WriteL(KSenDblQuot);
    }

EXPORT_C void CSenBaseElement::WriteAttrsToL(RWriteStream& aWriteStream)
    {
    for (TInt i = 0; i < iAttrs.Count(); i++)
        {
        WriteAttrToL(aWriteStream, iAttrs[i]->Name(), iAttrs[i]->Value());
        }
    }

EXPORT_C void CSenBaseElement::WriteNamespacesToL(RWriteStream& aWriteStream)
    {
    TInt count = iNamespaces.Count();
    CSenNamespace* ns = NULL;
    if (count > 0)
        {
        for (TInt i=0; i < count; i++)
            {
            ns = iNamespaces[i];
            if (ns)
                {
                aWriteStream.WriteL(KSenSpaceXmlns);
                if (ns->Prefix().Length() > 0)
                    {
                    aWriteStream.WriteL(KSenColon);
                    aWriteStream.WriteL(ns->Prefix());
                    }
                aWriteStream.WriteL(KSenEqualsDblQuot);
                aWriteStream.WriteL(ns->URI());
                aWriteStream.WriteL(KSenDblQuot);
                }
            }
        }
    }

EXPORT_C void CSenBaseElement::WriteElementsToL(RWriteStream& aWriteStream)
    {
    TInt elementCount(iElements.Count());
    for (TInt i=0; i<elementCount; i++)
        {
        iElements[i]->WriteAsXMLToL(aWriteStream);
        }
    }

EXPORT_C void CSenBaseElement::WriteContentToL(RWriteStream& aWriteStream)
    {
    aWriteStream.WriteL(Content());
    }

EXPORT_C MSenElement* CSenBaseElement::AsElement()
    {
    return this;
    }

EXPORT_C void CSenBaseElement::AllocContentBufL()
    {
    if(!ipContentBuf)
        {
        ipContentBuf = CBufFlat::NewL(KFlatBufSize);
        }
    }

EXPORT_C void CSenBaseElement::SetNamespaceL(const TDesC8& aNsUri)
    {
    SetNamespaceL(KNullDesC8, aNsUri);
    }

EXPORT_C void CSenBaseElement::SetNamespaceL(const TDesC8& aNsPrefix,
                                             const TDesC8& aNsUri)
    {
    if (aNsUri.Length() > 0)
        {
        ipNamespace = (CSenNamespace*) Namespace(aNsPrefix, aNsUri);
        if (!ipNamespace)
            {
            ipNamespace = CSenNamespace::NewL(aNsPrefix, aNsUri);

#ifdef EKA2
            iNamespaces.AppendL(ipNamespace);
#else
            User::LeaveIfError(iNamespaces.Append(ipNamespace));
#endif // EKA2
            }
        }
    }

EXPORT_C const CSenNamespace* CSenBaseElement::Namespace()
    {
    return ipNamespace;
    }

EXPORT_C const CSenNamespace* CSenBaseElement::Namespace(
                                                    const TDesC8& aNsPrefix)
    {
    return Namespace(aNsPrefix,ETrue);
    }

EXPORT_C const CSenNamespace* CSenBaseElement::Namespace(
                                                    const TDesC8& aNsPrefix,
                                                    const TBool aCheckInParent)
    {
    if (aNsPrefix.Length() == 0) return NULL;

    const CSenNamespace* pNamespace = NULL;

    TInt count(iNamespaces.Count());
    for (TInt i=0; i<count && pNamespace == NULL; i++)
        {
        pNamespace = iNamespaces[i];
        if (pNamespace)
            {
            if (pNamespace->Prefix() != aNsPrefix) pNamespace = NULL;
            }
        }

    if (pNamespace == NULL && ipParent && aCheckInParent)
        {
        pNamespace = ipParent->Namespace(aNsPrefix);
        }

    return pNamespace;
    }


EXPORT_C const CSenNamespace* CSenBaseElement::Namespace(
                                                    const TDesC8& aNsPrefix,
                                                    const TDesC8& aUri)
    {
    // If prefix == NULL, then any prefix is ok
    const CSenNamespace* pNs = NULL;
    // Check if the namespace has already been declared
    TInt count = iNamespaces.Count();
    if (count != 0)
        {
        for (TInt i=0; i<count; i++)
            {
            pNs = iNamespaces[i];
            if (!pNs->Compare(aNsPrefix, aUri))
                {
                pNs = NULL;
                }
            else
                {
                break;
                }
            }
        }
    if (pNs == NULL && ipParent != NULL)
        {
        // if no namespace defined, but there is a parent, ask its namespace
        pNs = ipParent->Namespace(aNsPrefix, aUri);
        }
    return pNs;
    }

EXPORT_C RPointerArray<CSenBaseAttribute>& CSenBaseElement::AttributesL()
    {
    return iAttrs;
    }

EXPORT_C RPointerArray<CSenNamespace>& CSenBaseElement::NamespacesL()
    {
    return iNamespaces;
    }

EXPORT_C void CSenBaseElement::CopyFromL(CSenElement& aSource)
    {
    TPtrC8 sourceContent = aSource.Content();
    if (sourceContent.Length() > 0)
        {
        if (ipContentBuf == NULL)
            {
            SetContentL(sourceContent);
            }
        else
            {
            RBufWriteStream bufWs(*ipContentBuf);
            CleanupClosePushL(bufWs);
            bufWs.WriteL(sourceContent);
            CleanupStack::PopAndDestroy(); // close bufWs
            }
        }

    RPointerArray<CSenNamespace> sourceNamespaces = aSource.NamespacesL();
    if (sourceNamespaces.Count() > 0)
        {
        for (TInt i=0;i<sourceNamespaces.Count(); i++)
            {
            CSenNamespace* pNamespace = sourceNamespaces[i];
            CSenNamespace* pNewNamespace =
                CSenNamespace::NewL(pNamespace->Prefix(),pNamespace->URI());
            CleanupStack::PushL(pNewNamespace);
#ifdef EKA2
            iNamespaces.AppendL(pNewNamespace);
#else
            User::LeaveIfError(iNamespaces.Append(pNewNamespace));
#endif

            CleanupStack::Pop(pNewNamespace);
            }
        }
    SetNamespaceL(aSource.NsPrefix(), aSource.NamespaceURI());

    RPointerArray<CSenBaseAttribute> sourceAttributes = aSource.AttributesL();
    if (sourceAttributes.Count() > 0)
        {
        for (TInt i=0;i<sourceAttributes.Count(); i++)
            {
            CSenBaseAttribute* pBaseAttribute = sourceAttributes[i];

            // duplicate check. Now overrides the original value
            CSenBaseAttribute* pOriginal = FindAttr(pBaseAttribute->Name());
            if (pOriginal)
                {
                pOriginal->SetValueL(pBaseAttribute->Value());
                continue;
                }

            CSenBaseAttribute* pNewBaseAttribute =
                CSenBaseAttribute::NewL(pBaseAttribute->Name(),
                                        pBaseAttribute->Value());
            CleanupStack::PushL(pNewBaseAttribute);
#ifdef EKA2
            iAttrs.AppendL(pNewBaseAttribute);
#else
            User::LeaveIfError(iAttrs.Append(pNewBaseAttribute));
#endif
            CleanupStack::Pop(pNewBaseAttribute);
            }
        }

    RPointerArray<CSenElement> sourceElements = aSource.ElementsL();
    if (sourceElements.Count() > 0)
        {
        for (TInt i=0;i<sourceElements.Count(); i++)
            {
            CSenElement* pElement = sourceElements[i];
            CSenElement* pNewElement =
                                CSenBaseElement::NewL(pElement->LocalName());
            pNewElement->SetParent(this);
            CleanupStack::PushL(pNewElement);
            pNewElement->CopyFromL(*pElement);
#ifdef EKA2
            iElements.AppendL(pNewElement);
#else
            User::LeaveIfError(iElements.Append(pNewElement));
#endif // EKA2
            CleanupStack::Pop(pNewElement);
            }
        }
    }

EXPORT_C void CSenBaseElement::SetPrefixL(const TDesC8& aPrefix)
    {
    if (iNamespaces.Find(ipNamespace) > -1)
        {
        //the namespace was locally declared

        // Check if element already has a namespace with given prefix
        CSenNamespace* pNamespace = NULL;
        TInt count(iNamespaces.Count());
        for (TInt i=0; i<count && pNamespace == NULL; i++)
            {
            pNamespace = iNamespaces[i];
            if (pNamespace->Prefix() != aPrefix) pNamespace = NULL;
            }

        if (pNamespace)
            {
            // Update existing namespace
            const TDesC8& uri = ipNamespace->URI();
            pNamespace->SetUriL(uri);
            ipNamespace = pNamespace;
            }
        else
            {
            if (ipNamespace->Prefix().Length() > 0 && iElements.Count() > 0)
                {
                // there may be children that depend on the old prefix in e.g. attribute names
                const TDesC8& uri = ipNamespace->URI();
                ipNamespace = CSenNamespace::NewL(aPrefix, uri);
#ifdef EKA2
                iNamespaces.AppendL(ipNamespace);
#else
                User::LeaveIfError(iNamespaces.Append(ipNamespace));
#endif
                }
            ipNamespace->SetPrefixL(aPrefix);
            }
        }
    }

EXPORT_C const CSenNamespace* CSenBaseElement::AddNamespaceL(
                                                CSenNamespace& aNewNamespace,
                                                TBool aCheckInParent)
    {
    const CSenNamespace* pNamespace =
                        Namespace(aNewNamespace.Prefix(), aCheckInParent);
    if (pNamespace == NULL)
        {
        CSenNamespace* pNewNamespace =
            CSenNamespace::NewL(aNewNamespace.Prefix(), aNewNamespace.URI());
        CleanupStack::PushL(pNewNamespace);
#ifdef EKA2
        iNamespaces.AppendL(pNewNamespace);
#else
        User::LeaveIfError(iNamespaces.Append(pNewNamespace));
#endif
        CleanupStack::Pop(); // pNewNamespace
        pNamespace = pNewNamespace;
        }
    return pNamespace;
    }

EXPORT_C const CSenNamespace* CSenBaseElement::AddNamespaceL(
                                                         const TDesC8& aPrefix,
                                                         const TDesC8& aUri)
    {
    CSenNamespace* pNamespace = (CSenNamespace*)Namespace(aPrefix);
		if (!pNamespace)
		{
    	if (aUri.Length() > 0)
        	{
        	pNamespace = CSenNamespace::NewL(aPrefix, aUri);
        	CleanupStack::PushL(pNamespace);
#ifdef EKA2
        	iNamespaces.AppendL(pNamespace);
#else
        	User::LeaveIfError(iNamespaces.Append(pNamespace));
#endif
        	CleanupStack::Pop(); // pNamespace
        	}
        }
	    else
	        {
	        pNamespace->SetUriL(aUri); 
	        }
		

    return pNamespace;
    }

EXPORT_C CSenElement* CSenBaseElement::DetachL()
    {
    if (!ipParent)
        {
        return NULL;
        }
    AddNamespaceMissingFromL(iNamespaces);
    ipParent->RemoveElement(*this);
    return this;
    }

void CSenBaseElement::AddNamespaceMissingFromL(
                                    RPointerArray<CSenNamespace>& aNamespaces)
    {
    // Add namespace of this element (possibly copied from parent)
    // if that namespace is not already defined in
    // aNamespaces array.
    if ( ipNamespace && aNamespaces.Find(ipNamespace) == -1)
        {
        // Check if namespace declaration of this element can
        // be found from aNamespaces array
        TInt count(aNamespaces.Count());
        TBool found(EFalse);
        TInt i = 0;
        for (; i<count; i++)
            {
            if ( aNamespaces[i]->URI() == ipNamespace->URI() && 
                 aNamespaces[i]->Prefix() == ipNamespace->Prefix() )
                {
                found = ETrue;
                break;
                }
            }
        if ( !found )
            {
            // Not found 
            // => Create a copy of namespace declaration
            //    and append it into aNamespaces array
            CSenNamespace* pCopy =
                    CSenNamespace::NewL(ipNamespace->Prefix(), ipNamespace->URI());
            CleanupStack::PushL(pCopy);
#ifdef EKA2
            aNamespaces.AppendL(pCopy);
#else
            User::LeaveIfError(aNamespaces.Append(pCopy));
#endif
            CleanupStack::Pop(pCopy);

            // Change namespace object for this element to namespace object
            // which can be found from aNamespaces array
            ipNamespace = pCopy;
            }
        else
            {
            // Change namespace object for this element to namespace object
            // which can be found from aNamespaces array
            
            ipNamespace = aNamespaces[i];
            }
        }

    // Add namespaces from this element's attributes
    // if those namespaces are not already defined in
    // aNamespaces array.
    if ( iAttrs.Count() > 0 )
        {
        TInt colon = -1;
        const CSenNamespace* pNamespace = NULL;

        TInt count(iAttrs.Count());
        for (TInt i=0; i<count; i++)
            {
            CSenBaseAttribute* pAttribute = iAttrs[i];

            // Check namespace assigned to Name
            colon = pAttribute->Name().Locate(':');
            if ( colon > 0 )
                {
                pNamespace =
                    Namespace(pAttribute->Name().Mid(0, colon), KNullDesC8);
                }
            if ( pNamespace && (aNamespaces.Find(pNamespace) == -1) )
                {
                // Check if namespace declaration for handled attribute
                // _name_ can be found from aNamespaces array.
                TInt count(aNamespaces.Count());
                TBool found(EFalse);
                for (TInt i = 0; i<count; i++)
                    {
                    if ( aNamespaces[i]->URI() == pNamespace->URI() && 
                         aNamespaces[i]->Prefix() == pNamespace->Prefix() )
                        {
                        found = ETrue;
                        break;
                        }
                    }
                if ( !found )
                    {
                    // Not found 
                    // => Create a copy of namespace declaration
                    //    and append it into aNamespaces array
                    CSenNamespace* pCopy =
                            CSenNamespace::NewL(pNamespace->Prefix(),
                                                pNamespace->URI());
                    CleanupStack::PushL(pCopy);
#ifdef EKA2
                    aNamespaces.AppendL(pCopy);
#else
                    User::LeaveIfError(aNamespaces.Append(pCopy));
#endif
                    CleanupStack::Pop(pCopy);
                    }
                }
            pNamespace = NULL;

            // Check namespace assigned to Value
            colon = pAttribute->Value().Locate(':');
            if ( colon > 0 )
                {
                pNamespace =
                    Namespace(pAttribute->Value().Mid(0, colon), KNullDesC8);
                }
            if ( pNamespace && (aNamespaces.Find(pNamespace) == -1) )
                {
                // Check if namespace declaration for handled attribute
                // _value_ can be found from aNamespaces array.
                TInt count(aNamespaces.Count());
                TBool found(EFalse);
                for (TInt i = 0; i<count; i++)
                    {
                    if ( aNamespaces[i]->URI() == pNamespace->URI() && 
                         aNamespaces[i]->Prefix() == pNamespace->Prefix() )
                        {
                        found = ETrue;
                        break;
                        }
                    }
                if ( !found )
                    {
                    // Not found 
                    // => Create a copy of namespace declaration
                    //    and append it into aNamespaces array
                    CSenNamespace* pCopy =
                            CSenNamespace::NewL(pNamespace->Prefix(), 
                                                pNamespace->URI());
                    CleanupStack::PushL(pCopy);
#ifdef EKA2
                    aNamespaces.AppendL(pCopy);
#else
                    User::LeaveIfError(aNamespaces.Append(pCopy));
#endif
                    CleanupStack::Pop(pCopy);
                    }
                }
            pNamespace = NULL;
            }
        }

    // Do recursive search for all child elements and for all child
    // elements of all child elements and so on.
    if ( iElements.Count() > 0 )
        {
        TInt count(iElements.Count());
        for (TInt i=0; i<count; i++)
            {
            CSenBaseElement* pElement = (CSenBaseElement*) iElements[i];
            pElement->AddNamespaceMissingFromL(aNamespaces);
            }
        }
    }


EXPORT_C void CSenBaseElement::Set(const TDesC8& aNamespaceURI,
                                   const TDesC8& aLocalName,
                                   const TDesC8& aQName)
    {
    
    HBufC8* pTemp = aLocalName.Alloc();
    if(pTemp)
        {
        delete ipLocalName;
        ipLocalName = NULL;
        ipLocalName = pTemp;
        }


    TPtrC8 prefix(KNullDesC8);

    if(aQName != KNullDesC8)
        {
        TInt colon(KErrNotFound);
        colon = aQName.Locate(':');
        if(colon!=KErrNotFound)
            {
            prefix.Set(aQName.Left(colon));
            }
        }
    TInt leaveCode(KErrNone);
    TRAP(leaveCode, SetNamespaceL(prefix, aNamespaceURI);)
    leaveCode=0;
    
    //if (err) ; 
    }

EXPORT_C CSenElement* CSenBaseElement::Child(TInt aIndex)
    {
    CSenElement* pElement = NULL;
    if (aIndex < iElements.Count())
        {
        pElement = (CSenBaseElement*) iElements[aIndex];
        }
    return pElement;
    }

EXPORT_C TBool CSenBaseElement::ConsistsOfL(MSenElement& aCandidate)
    {
    // First check the names and namespaces
    if (    aCandidate.LocalName() != this->LocalName()
        ||  aCandidate.NamespaceURI() != this->NamespaceURI()
        ||  aCandidate.NsPrefix() != this->NsPrefix()
        )
        {
        return EFalse;
        }

    // Then check content if it exists
    if (aCandidate.HasContent())
        {
        if (aCandidate.Content() != this->Content())
            {
            return EFalse;  // Content doesn't match => no match
            }
        }

    // Then handle the children
    RPointerArray<CSenElement>& children = aCandidate.ElementsL();
    TInt childCount = children.Count();
    TInt i=0;
    while (i < childCount)
        {
        CSenElement* pMatchChild = children[i];
        CSenElement* pChild = this->Element(pMatchChild->LocalName());
        if (!pChild)
            {
            return EFalse;  // no child with same name found in children, 
                            // => no match
            }
        else
            {
            if (!pChild->ConsistsOfL(*pMatchChild))
                {
                return EFalse; // a non matching child was found => No match
                }
            }
        i++;
        }
    return ETrue;
    }

// End of File