|
Hello!
Here are a few more functions, perhaps useful to others. First off, I would suggest making all accessors const, and I also changed CString to std::string. For me, this was more convenient. Anyway, here are the functions:
<code>
string CXmlNodeWrapper::GetNamespaceURI() const
{
if( IsValid() )
{
_bstr_t str = static_cast<_bstr_t>(m_xmlnode->GetnamespaceURI());
return static_cast<char*>(str);
}
return "";
}
string CXmlNodeWrapper::GetNodeValue() const
{
if( IsValid() ) {
_bstr_t str = static_cast<_bstr_t>(m_xmlnode->GetnodeValue());
return static_cast<char*>(str);
}
return "";
}
MSXML2::IXMLDOMNodePtr CXmlNodeWrapper::InsertNode(int index, LPCTSTR nodeName, LPCTSTR nameSpace)
{
MSXML2::IXMLDOMDocumentPtr xmlDocument = m_xmlnode->GetownerDocument();
if (xmlDocument)
{
MSXML2::IXMLDOMNodePtr newNode = xmlDocument->createNode(_variant_t((short)MSXML2::NODE_ELEMENT), nodeName, nameSpace);
MSXML2::IXMLDOMNodePtr refNode = GetNode(index);
if (refNode)
newNode = m_xmlnode->insertBefore(newNode,_variant_t(refNode.GetInterfacePtr()));
else
newNode = m_xmlnode->appendChild(newNode);
return newNode;
}
return NULL;
}
void CXmlDocumentWrapper::CreateRootElement(LPCSTR name, LPCTSTR nameSpace)
{
MSXML2::IXMLDOMElementPtr pe;
pe = m_xmldoc->createNode("element", name, nameSpace);
if( pe!=NULL )
{
m_xmldoc->appendChild(pe);
pe.Release();
}
}
string CXmlDocumentWrapper::GetParseError() const
{
if( IsValid() )
{
MSXML2::IXMLDOMParseErrorPtr error = m_xmldoc->GetparseError();
boost::format f("Error:\nCode = 0x%1$x\nSource = Line: %2%, Char: %3%\nError Description = %4%");
_bstr_t bstrErr(error->Getreason());
f % error->GeterrorCode() % error->Getline() % error->Getlinepos() % (char*)bstrErr;
return f.str();
}
else
return "";
}
void CXmlDocumentWrapper::CreateProcessingInstruction(LPCTSTR target, LPCTSTR data)
{
MSXML2::IXMLDOMProcessingInstructionPtr pi;
pi = m_xmldoc->createProcessingInstruction(target, data);
if( pi!=NULL )
{
m_xmldoc->appendChild(pi);
pi.Release();
}
}
</code>
That's it.
Thanks for a great set of wrappers!
Daniel Lidström
|
|
|
|
|
Thanks Daniel.
I'm very happy to see that someone is not only blindly using the the wrapper (although that is perfectly OK , that was the intention), but also developing and farther enriching it.
Again , thank you very much.
Creative minds - create creative creations!
|
|
|
|
|
Hi Alex,
Why won't you update your excellent class with those ammenments then?
I'm observing it now to see how it will fit to my current application. If it will, you will absolutely get a 5!
Stilgar.
|
|
|
|
|
Sorry but I don't know how to create an xml like
<?xml verion = "1.0" encoding = "UTF-8"?>
<root>
<attb name = "1">
<ele name = "1.1">
</attb>
<attb name = "2">
<ele name = "2.1">
</attb>
</root>
I called:
doc.CreateProcessingInstruction("xml", "1.0");
doc.LoadXML("<root></root>");
...
but it didn't at <?xml = "1.0"?> at the beginning of the doc
I called:
doc.LoadXML("<root></root>");
doc.CreateProcessingInstruction("xml", "1.0");
...
it insert <?xml = "1.0"?> after <root> element.
Please help! Thank you in advanced!
|
|
|
|
|
Hello!
How do I access the node value of a text node. For example, the xml is this:
<?xml version="1.0" encoding="utf-8"?>
<Model mod_att1="val1" mod_att2="val2">__model_value__
<MainLine att1="val1" att2="val2">__mainline_value__
</MainLine>
</Model>
If I have a CXmlNodeWrapper pointing to the top element, Model, and I call GetText, then I receive a string with a value of
"__model_value__
__mainline_value__"
How do I get the value of the Model element? I'm only interested in the __model_value__ part.
Thanks in advance!
|
|
|
|
|
I figured it out. Add a method to CXmlNodeWrapper like this:
class CXmlNodeWrapper
{
...
//! Access the value of the node
std::string GetNodeValue() const;
};
string CXmlNodeWrapper::GetNodeValue() const
{
if( IsValid() ) {
_bstr_t str = static_cast<_bstr_t>(m_xmlnode->GetnodeValue());
return static_cast<char*>(str);
}
return "";
}
This method may only be called when NodeType() returns "text".
|
|
|
|
|
Hello :->
As I can't find a solution for my GetUrl() problem, I changed my objective :
I would like to use the "Serialize" function to save my DOM....like that, I don't care of all open/save functions
Is it possible to serialize my DOM ?
Bye.
|
|
|
|
|
I am going to use MSXML 3.0 in my project. I have a time critical process where i need to generate a big XML file within 3 sec This is the maximum amount of time avaiable for the process to complete.Can MSXML 3.0 do it for me. Anybody any comments.
|
|
|
|
|
Of course it can, it has only about 10000000 different "ifs".
Such as what prosessor do you have, what is the speed of your hard disk, wether your process has the highest priority , what processor time is available for your process, how big is your file going to be, is the mashine doing something elese at this moment (updating the untivirus for example) , etc...
What I'm trying to say is that you should NEVER under no circumstanses expect some task finish in some predefined time , please either work on some other strategy in your application or at least explain why is the time so critical.
Creative minds - create creative creations!
|
|
|
|
|
Why the process is time critical ? the answer to that is , i am generating an export for a device configuration , the configuration is saved as an xml file . Now when the user exports the configuration , i dont have any status bar for indicating the status of the export, thus the application which i am using for export appears to be hung. I dont want to spoil the user experience by taking too long a time to generate an export,thats why a self imposed time criticality is there.I was talking to one of my friends about MSXML 3.0 taking too long to taverse the DOM tree if the XML document is too big , I wanted your opinion on weather MSXML is slow . I have worked with it earlier and i found it really fast , generating an xml in milli seconds.But i still need a confiration from someone before i go ahead.After reading the article i felt you might be the one. Thanx in advance for paying attention.
|
|
|
|
|
MSXML is capble of producing 50000 nodes XML in less than a second.
It also capble of quiering this XML 5~50 quieries in less than a second.
But as I said before it all depends on many issues.
My advise for you is to try first, than if you see that the process of saving the configuration is taking too long (more than couple of seconds) consined adding a progress bar or if that is not possible, at least some flashing message like "Saving configuration, please wait...".
Creative minds - create creative creations!
|
|
|
|
|
Hi,
first : thanks for your good work.
I'm trying to use your function GetUrl() to know if the document has been already saved, but it doesn't work.
<br />
if(_myDoc.GetUrl() != ""){<br />
return (_myDoc.Save(""));<br />
}else{<br />
CFileDialog fDlg(FALSE,"xml",NULL,OFN_HIDEREADONLY | OFN_PATHMUSTEXIST,"Xml Files (*.xml)|*.xml||");<br />
...<br />
Can you give me a clue ?
Thx !
|
|
|
|
|
What is it exactly that doesn't work?
What do you get from the GetUrl function?
Creative minds - create creative creations!
|
|
|
|
|
I always get an empty string...and that's strange because always use Load or Save before.
Debug-mode says that returned PTR for url is invalid and converts it to "" (???)
Thx for help
|
|
|
|
|
Hi~ Alex:
First I would like to thank you for your wrapper class. It is a perfect class and help me a lot.
However, I have one little question that need you help to answer.
1. I had used your class and I can add a new node in xml file. But when I open the xml by text editor like UltraEdit. It did not have the line feed effection.
For example:
////////////////////////////////////////////////
// original
<book id="bk112">
<author>Galos, Mike
<title>Visual Studio 7: A Comprehensive Guide
<genre>Computer
<price>49.95
<publish_date>2001-04-16
<description>Microsoft Visual Studio 7 is explored in depth, looking at how Visual Basic, Visual C++, C#, and ASP+ are integrated into a comprehensive development environment.
////////////////////////////////////////////////
// New item
<book_user id="123"><arthur>Keven Yang<title>This is a test<grene>test<date>20050728<description>This is a test item
If you have free time, please help me to answer it!
thanks...
|
|
|
|
|
You should alter the "preserveWhiteSpace" property of DOMDocument.
This requires you to change the originginal code!
Creative minds - create creative creations!
|
|
|
|
|
Hi, I’m new to using XML but would like to incorporate it into one of the applications I’m writing. I am very clueless about both XML in general and using this class. I’ve painfully tried many things and just can’t seem to get the output I’m looking for.
First of all, here is an XML example of what I’m trying to do:
<PrinterDefines>
<Group>Epson SIDM Printers
<Printer>LQ-570
<Lang>ESC/P2</Lang>
<Pins>24</Pins>
</Printer>
<Printer>FX-870
<Lang>ESC/P2</Lang>
<Pins>9</Pins>
</Printer>
</Group>
<Group>Epson POS Printers
<Printer>TM-U220
<Lang>ESC/POS</Lang>
<Pins>9</Pins>
</Printer>
</Group>
</PrinterDefines>
Ideally, I’m defining printer groups and types. The groups in this example are Epson SIDM printers and Epson POS printers. Under each group, there are printers listed. For example, under the SIDM group, there are two printers defined: LQ-570 and FX-870. Under each printer there are further fields pertaining to that printer. For example, the LQ-570 uses ESC/P2 language and it has 24 pins.
What I’m trying to do is obviously retrieve this data. For example, if I want to retrieve the LQ-570, I also want to retrieve it’s language and the number of pins associated with it; and I want these individual strings to end-up in CStrings. Of course, these are two of many fields that I need to associate with that printer. This one is just a start so that I can understand it. Once I understand it, the list will be quite complex, and this data will end up in various classes. For now, CStrings will be fine for me to learn.
Anyway, I can successfully get the entire list of ‘Groups’ with the following:
CXmlNodeListWrapper GroupList(node.FindNodes("//Group"));
My problem is I don’t know what to do next! I’ve tried many things, and each is way way off, so any help would greatly be appreciated.
Thanks!
|
|
|
|
|
To get specific group:
CXmlNodeWrapper Group(node.FindNode("//Group[text() = 'Epson POS Printers'"));
This will get the group "Epson POS Printers"
To get specific printer:
CXmlNodeWrapper Printer(node.FindNode("//Printer[text() = 'FX-870'"));
This will get the FX-870 printer.
Since all the printer properties are defined as Nodes rather than attributes,
You have to use GetNode function to get a specific node, and you cant use GetValue function.
Example:
CXmlNodeWrapper node;
node = Printer.GetNode("Lang");
CString szLang = node.Text();
Creative minds - create creative creations!
|
|
|
|
|
Hi Alex, Thanks for the info. In my case, the example to get a specific group such as:
CXmlNodeWrapper Group(node.FindNode("//Group[text() = 'Epson POS Printers'"));
will not work for me as I am defining the groups and printers in the XML file. In other words, I do not know the names to search for until I've loaded and parsed the XML file. The XML file defines printer types, names, and properties in my application. Anyway, here is the way I ended up doing it and it worked fine for me...
void CPrinterFileInfo::ParsePrinterDefinesNode(IDispatch *pNode)
{
CGroupEntryInfo *gei;
CXmlNodeWrapper node(pNode);
CXmlNodeListWrapper GroupList(node.FindNodes(_T("//Group"))); // Thermal, SIDM, etc...
GroupList.Start();
CXmlNodeWrapper List2(GroupList.Next());
while(List2.IsValid())
{
gei = new CGroupEntryInfo;
ParseGroupDefinesNode(List2, *(gei));
AddGroupEntry(*(gei));
List2 = GroupList.Next();
delete gei;
}
}
void CPrinterFileInfo::ParseGroupDefinesNode(CXmlNodeWrapper& node, CGroupEntryInfo& gei)
{
CDeviceEntryInfo *dei;
gei.SetGroupName(node.GetAttribVal(XML_GROUP_NAME));
gei.SetGroupDescription(node.GetAttribVal(XML_GROUP_DESC));
for(int j= 0; j < node.NumNodes(); j++)
{
dei = new CDeviceEntryInfo;
CXmlNodeWrapper List(node.GetNode(j));
ParseMemberDefinesNode(List, *(dei));
gei.AddMemberEntry(*(dei));
delete dei;
}
}
void CPrinterFileInfo::ParseMemberDefinesNode(CXmlNodeWrapper& node, CDeviceEntryInfo& dei)
{
dei.SetDeviceName(node.GetAttribVal(XML_DEVICE_NAME));
dei.SetDeviceClass(node.GetAttribVal(XML_DEVICE_CLASS));
dei.SetDeviceFeatures(node.GetAttribVal(XML_DEVICE_FEATURES));
dei.SetDevicePaperPaths(node.GetAttribVal(XML_DEVICE_PAPER));
dei.SetDeviceFonts(node.GetAttribVal(XML_DEVICE_FONTS));
dei.SetDeviceColumns(node.GetAttribVal(XML_DEVICE_COLS));
dei.SetDeviceInterface(node.GetAttribVal(XML_DEVICE_INTERFACES));
dei.SetDeviceLanguage(node.GetAttribVal(XML_DEVICE_LANG));
dei.SetDeviceExFeatures(node.GetAttribVal(XML_DEVICE_EXFEATURES));
dei.SetDeviceBuffer(node.GetAttribVal(XML_DEVICE_BUFFERSIZE));
}
Part of the XML File...
<?xml version="1.0" encoding="UTF-8"?>
<DeviceDefines>
<Group Name="TM-T Series" Desc="Thermal Point Of Sale Printers">
<Device Name="TM-T88III"
Class="1"
Features="2147610751"
PaperPaths="15"
Fonts="3"
Cols="12"
Interface="15"
Lang="3"
ExFeatures="41"
Buffer="4096">
</Device>
</Group>
</DeviceDefines>
Thanks for your reply!
|
|
|
|
|
I have two CXmlDocumentWrapper objects (doc1 and doc2). And I need to copy xml document from doc1 to doc2. So I must free memory used by doc1.m_xmldoc and copy data from doc2.m_xmldoc to doc1.m_xmldoc. As I can see now you working with pointers, so if I change doc1.m_xmldoc, doc2.m_xmldoc will be changed too. What should I do?
I cannot use Clone method because I must free doc1.m_xmldoc first
|
|
|
|
|
Assume you have doc1 object:
CXmlDocumentWrapper doc2(doc1.Clone());
Creative minds - create creative creations!
|
|
|
|
|
I mean if I have some xml document already loaded in doc2
CXmlDocumentWrapper doc2;
doc2.Load( _T("sample.xml") );
....
doc2 = doc1.Clone();
So I want to know will the memory already occupied in doc2 be free when I do doc2 = doc1.Clone() ?
|
|
|
|
|
Yep.
Creative minds - create creative creations!
|
|
|
|
|
Hi, Alex
I have only one little question: How I can add the string like <?xml version="1.0" encoding="UTF-8"?> in my xml file. I try to do like your suggestion in your old article( insert createProcessingInstruction() in the document constructor). But in this case if I make after then LoadXML(something) to start new tree, the instruction row has been disappeared.
I fixed it now by loading doc from template file, but may be you have a better resolve?
Thanks
|
|
|
|
|
The create processing instructio function is not implemented in this release, but since the class operates on ditect pointers you can ask for raw pointer to the document via Interface() finction and add the instruction yourself.
Creative minds - create creative creations!
|
|
|
|