This repository has been archived on 2025-12-24. You can view files and clone it, but you cannot make any changes to it's state, such as pushing and creating new issues, pull requests or comments.
yachat/iris-legacy/iris/xmpp-im/xmpp_xdata.cpp

474 lines
9.9 KiB
C++
Raw Permalink Normal View History

2025-12-25 01:37:49 +05:00
/*
* xmpp_xdata.cpp - a class for jabber:x:data forms
* Copyright (C) 2003-2004 Michail Pishchagin
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#include "xmpp_xdata.h"
#include "xmpp_xmlcommon.h"
#include "xmpp_jid.h"
#include <QList>
#include <QSharedDataPointer>
using namespace XMPP;
using namespace XMLHelper;
// TODO: report, item
XData::Private::Private()
: QSharedData()
{
type = Data_Form;
}
//----------------------------------------------------------------------------
// XData::Field
//----------------------------------------------------------------------------
XData::Field::Field()
{
}
XData::Field::~Field()
{
}
QString XData::Field::desc() const
{
return _desc;
}
void XData::Field::setDesc(const QString &d)
{
_desc = d;
}
XData::Field::OptionList XData::Field::options() const
{
return _options;
}
void XData::Field::setOptions(XData::Field::OptionList o)
{
_options = o;
}
bool XData::Field::required() const
{
return _required;
}
void XData::Field::setRequired(bool r)
{
_required = r;
}
QString XData::Field::label() const
{
return _label;
}
void XData::Field::setLabel(const QString &l)
{
_label = l;
}
QString XData::Field::var() const
{
return _var;
}
void XData::Field::setVar(const QString &v)
{
_var = v;
}
QStringList XData::Field::value() const
{
return _value;
}
void XData::Field::setValue(const QStringList &v)
{
_value = v;
}
XData::Field::Type XData::Field::type() const
{
return _type;
}
void XData::Field::setType(XData::Field::Type t)
{
_type = t;
}
bool XData::Field::isValid() const
{
if ( _required && _value.isEmpty() )
return false;
if ( _type == Field_Hidden || _type == Field_Fixed) {
return true;
}
if ( _type == Field_Boolean ) {
if ( _value.count() != 1 )
return false;
QString str = _value.first();
if ( str == "0" || str == "1" || str == "true" || str == "false" || str == "yes" || str == "no" )
return true;
}
if ( _type == Field_TextSingle || _type == Field_TextPrivate ) {
if ( _value.count() == 1 )
return true;
}
if ( _type == Field_TextMulti ) {
//no particular test. empty/required case already caught (see above)
return true;
}
if ( _type == Field_ListSingle || _type == Field_ListMulti ) {
//no particular test. empty/required case already caught (see above)
return true;
}
if ( _type == Field_JidSingle ) {
if ( _value.count() != 1 )
return false;
Jid j( _value.first() );
return j.isValid();
}
if ( _type == Field_JidMulti ) {
QStringList::ConstIterator it = _value.begin();
bool allValid = true;
for ( ; it != _value.end(); ++it) {
Jid j(*it);
if ( !j.isValid() ) {
allValid = false;
break;
}
}
return allValid;
}
return false;
}
void XData::Field::fromXml(const QDomElement &e)
{
if ( e.tagName() != "field" )
return;
_var = e.attribute("var");
_label = e.attribute("label");
QString type = e.attribute("type");
if ( type == "boolean" )
_type = Field_Boolean;
else if ( type == "fixed" )
_type = Field_Fixed;
else if ( type == "hidden" )
_type = Field_Hidden;
else if ( type == "jid-multi" )
_type = Field_JidMulti;
else if ( type == "jid-single" )
_type = Field_JidSingle;
else if ( type == "list-multi" )
_type = Field_ListMulti;
else if ( type == "list-single" )
_type = Field_ListSingle;
else if ( type == "text-multi" )
_type = Field_TextMulti;
else if ( type == "text-private" )
_type = Field_TextPrivate;
else
_type = Field_TextSingle;
_required = false;
_desc = QString::null;
_options.clear();
_value.clear();
QDomNode n = e.firstChild();
for ( ; !n.isNull(); n = n.nextSibling() ) {
QDomElement i = n.toElement();
if ( i.isNull() )
continue;
QString tag = i.tagName();
if ( tag == "required" )
_required = true;
else if ( tag == "desc" )
_desc = i.text().simplifyWhiteSpace();
else if ( tag == "option" ) {
Option o;
bool found;
o.label = i.attribute("label");
QDomElement e = findSubTag( i, "value", &found );
o.value = ( found ? e.text() : QString("") );
_options.append(o);
}
else if ( tag == "value" ) {
_value.append(i.text());
}
}
}
QDomElement XData::Field::toXml(QDomDocument *doc, bool submitForm) const
{
QDomElement f = doc->createElement("field");
// setting attributes...
if ( !_var.isEmpty() )
f.setAttribute("var", _var);
if ( !submitForm && !_label.isEmpty() )
f.setAttribute("label", _label);
// now we're gonna get the 'type'
QString type = "text-single";
if ( _type == Field_Boolean )
type = "boolean";
else if ( _type == Field_Fixed )
type = "fixed";
else if ( _type == Field_Hidden )
type = "hidden";
else if ( _type == Field_JidMulti )
type = "jid-multi";
else if ( _type == Field_JidSingle )
type = "jid-single";
else if ( _type == Field_ListMulti )
type = "list-multi";
else if ( _type == Field_ListSingle )
type = "list-single";
else if ( _type == Field_TextMulti )
type = "text-multi";
else if ( _type == Field_TextPrivate )
type = "text-private";
f.setAttribute("type", type);
// now, setting nested tags...
if ( !submitForm && _required )
f.appendChild( emptyTag(doc, "required") );
if ( !submitForm && !_desc.isEmpty() )
f.appendChild( textTag(doc, "desc", _desc) );
if ( !submitForm && !_options.isEmpty() ) {
OptionList::ConstIterator it = _options.begin();
for ( ; it != _options.end(); ++it) {
QDomElement o = doc->createElement("option");
o.appendChild(textTag(doc, "value", (*it).value));
if ( !(*it).label.isEmpty() )
o.setAttribute("label", (*it).label);
f.appendChild(o);
}
}
if ( !_value.isEmpty() ) {
QStringList::ConstIterator it = _value.begin();
for ( ; it != _value.end(); ++it)
f.appendChild( textTag(doc, "value", *it) );
}
return f;
}
//----------------------------------------------------------------------------
// XData
//----------------------------------------------------------------------------
XData::XData()
{
d = new Private;
}
QString XData::title() const
{
return d->title;
}
void XData::setTitle(const QString &t)
{
d->title = t;
}
QString XData::instructions() const
{
return d->instructions;
}
void XData::setInstructions(const QString &i)
{
d->instructions = i;
}
XData::Type XData::type() const
{
return d->type;
}
void XData::setType(Type t)
{
d->type = t;
}
XData::FieldList XData::fields() const
{
return d->fields;
}
void XData::setFields(const FieldList &f)
{
d->fields = f;
}
void XData::fromXml(const QDomElement &e)
{
if ( (e.attribute("xmlns") != "jabber:x:data") && (e.namespaceURI() != "jabber:x:data") )
return;
QString type = e.attribute("type");
if ( type == "result" )
d->type = Data_Result;
else if ( type == "submit" )
d->type = Data_Submit;
else if ( type == "cancel" )
d->type = Data_Cancel;
else
d->type = Data_Form;
d->title = subTagText(e, "title");
d->instructions = subTagText(e, "instructions");
d->fields.clear();
QDomNode n = e.firstChild();
for ( ; !n.isNull(); n = n.nextSibling() ) {
QDomElement i = n.toElement();
if ( i.isNull() )
continue;
if ( i.tagName() == "field" ) {
Field f;
f.fromXml(i);
d->fields.append(f);
}
else if ( i.tagName() == "reported" ) {
d->report.clear();
d->reportItems.clear();
QDomNode nn = i.firstChild();
for ( ; !nn.isNull(); nn = nn.nextSibling() ) {
QDomElement ii = nn.toElement();
if ( ii.isNull() )
continue;
if ( ii.tagName() == "field" ) {
d->report.append( ReportField( ii.attribute("label"), ii.attribute("var") ) );
}
}
}
else if ( i.tagName() == "item" ) {
ReportItem item;
QDomNode nn = i.firstChild();
for ( ; !nn.isNull(); nn = nn.nextSibling() ) {
QDomElement ii = nn.toElement();
if ( ii.isNull() )
continue;
if ( ii.tagName() == "field" ) {
QString name = ii.attribute("var");
QString value;
bool found;
QDomElement e = findSubTag( ii, "value", &found );
if ( found )
value = e.text();
item[name] = value;
}
}
d->reportItems.append( item );
}
}
}
QDomElement XData::toXml(QDomDocument *doc, bool submitForm) const
{
QDomElement x = doc->createElementNS("jabber:x:data", "x");
x.setAttribute("xmlns", "jabber:x:data");
QString type = "form";
if ( d->type == Data_Result )
type = "result";
else if ( d->type == Data_Submit )
type = "submit";
else if ( d->type == Data_Cancel )
type = "cancel";
x.setAttribute("type", type);
if ( !submitForm && !d->title.isEmpty() )
x.appendChild( textTag(doc, "title", d->title) );
if ( !submitForm && !d->instructions.isEmpty() )
x.appendChild( textTag(doc, "instructions", d->instructions) );
if ( !d->fields.isEmpty() ) {
FieldList::ConstIterator it = d->fields.begin();
for ( ; it != d->fields.end(); ++it) {
Field f = *it;
if ( !(submitForm && f.var().isEmpty()) )
x.appendChild( f.toXml(doc, submitForm) );
}
}
return x;
}
const QList<XData::ReportField> &XData::report() const
{
return d->report;
}
const QList<XData::ReportItem> &XData::reportItems() const
{
return d->reportItems;
}
bool XData::isValid() const
{
foreach(Field f, d->fields) {
if (!f.isValid())
return false;
}
return true;
}
void XData::addField(XMPP::XData::Field::Type type, const QString& var, const QString& value)
{
XMPP::XData::Field field;
field.setVar(var);
field.setType(type);
field.setValue(QStringList() << value);
d->fields << field;
}