summaryrefslogtreecommitdiffstats
path: root/src/corelib/io/qurlquery.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/corelib/io/qurlquery.cpp')
-rw-r--r--src/corelib/io/qurlquery.cpp745
1 files changed, 745 insertions, 0 deletions
diff --git a/src/corelib/io/qurlquery.cpp b/src/corelib/io/qurlquery.cpp
new file mode 100644
index 0000000000..2f1975722f
--- /dev/null
+++ b/src/corelib/io/qurlquery.cpp
@@ -0,0 +1,745 @@
+/****************************************************************************
+**
+** Copyright (C) 2012 Intel Corporation
+** Contact: http://www.qt-project.org/
+**
+** This file is part of the QtCore module of the Qt Toolkit.
+**
+** $QT_BEGIN_LICENSE:LGPL$
+** GNU Lesser General Public License Usage
+** This file may be used under the terms of the GNU Lesser General Public
+** License version 2.1 as published by the Free Software Foundation and
+** appearing in the file LICENSE.LGPL included in the packaging of this
+** file. Please review the following information to ensure the GNU Lesser
+** General Public License version 2.1 requirements will be met:
+** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** In addition, as a special exception, Nokia gives you certain additional
+** rights. These rights are described in the Nokia Qt LGPL Exception
+** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
+**
+** GNU General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU General
+** Public License version 3.0 as published by the Free Software Foundation
+** and appearing in the file LICENSE.GPL included in the packaging of this
+** file. Please review the following information to ensure the GNU General
+** Public License version 3.0 requirements will be met:
+** http://www.gnu.org/copyleft/gpl.html.
+**
+** Other Usage
+** Alternatively, this file may be used in accordance with the terms and
+** conditions contained in a signed written agreement between you and Nokia.
+**
+**
+**
+**
+**
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+#include "qurlquery.h"
+
+QT_BEGIN_NAMESPACE
+
+/*!
+ \class QUrlQuery
+
+ \brief The QUrlQuery class provides a way to manipulate a key-value pairs in
+ a URL's query.
+
+ \reentrant
+ \ingroup io
+ \ingroup network
+ \ingroup shared
+
+ It is used to parse the query strings found in URLs like the following:
+
+ \img qurl-querystring.png
+
+ Query strings like the above are used to transmit options in the URL and are
+ usually decoded into multiple key-value pairs. The one above would contain
+ two entries in its list, with keys "type" and "color". QUrlQuery can also be
+ used to create a query string suitable for use in QUrl::setQuery() from the
+ individual components of the query.
+
+ The most common way of parsing a query string is to initialize it in the
+ constructor by passing it the query string. Otherwise, the setQuery() method
+ can be used to set the query to be parsed. That method can also be used to
+ parse a query with non-standard delimiters, after having set them using the
+ setQueryDelimiters() function.
+
+ The encoded query string can be obtained again using query(). This will take
+ all the internally-stored items and encode the string using the delimiters.
+
+ \section1 Encoding
+
+ All of the getter methods in QUrlQuery support an optional parameter of type
+ QUrl::ComponentFormattingOptions, including query(), which dictate how to
+ encode the data in question. Regardless of the mode, the returned value must
+ still be considered a percent-encoded string, as there are certain values
+ which cannot be expressed in decoded form (like control characters, byte
+ sequences not decodable to UTF-8). For that reason, the percent character is
+ always represented by the string "%25".
+
+ \section2 Handling of spaces and plus ("+")
+
+ Web browsers usually encode spaces found in HTML FORM elements to a plus sign
+ ("+") and plus signs to its percent-encoded form (%2B). However, the Internet
+ specifications governing URLs do not consider spaces and the plus character
+ equivalent.
+
+ For that reason, QUrlQuery never encodes the space character to "+" and will
+ never decode "+" to a space character. Instead, space characters will be
+ rendered "%20" in encoded form.
+
+ To support encoding like that of HTML forms, QUrlQuery also never decodes the
+ "%2B" sequence to a plus sign nor encode a plus sign. In fact, any "%2B" or
+ "+" sequences found in the keys, values, or query string are left exactly
+ like written (except for the uppercasing of "%2b" to "%2B").
+
+ \section1 Non-standard delimiters
+
+ By default, QUrlQuery uses an equal sign ("=") to separate a key from its
+ value, and an ampersand ("&") to separate key-value pairs from each other. It
+ is possible to change the delimiters that QUrlQuery uses for parsing and for
+ reconstructing the query by calling setQueryDelimiters().
+
+ Non-standard delimiters should be chosen from among what RFC 3986 calls
+ "sub-delimiters". They are:
+
+ \code
+ sub-delims = "!" / "$" / "&" / "'" / "(" / ")"
+ / "*" / "+" / "," / ";" / "="
+ \endcode
+
+ Use of other characters is not supported and may result in unexpected
+ behaviour. QUrlQuery does not verify that you passed a valid delimiter.
+
+ \sa QUrl
+*/
+
+typedef QList<QPair<QString, QString> > Map;
+
+int qt_urlRecode(QString &appendTo, const QChar *begin, const QChar *end,
+ QUrl::ComponentFormattingOptions encoding, const ushort *tableModifications);
+
+class QUrlQueryPrivate : public QSharedData
+{
+public:
+ QUrlQueryPrivate(const QString &query = QString())
+ : valueDelimiter(QUrlQuery::defaultQueryValueDelimiter()),
+ pairDelimiter(QUrlQuery::defaultQueryPairDelimiter())
+ { if (!query.isEmpty()) setQuery(query); }
+
+ QString recodeFromUser(const QString &input) const;
+ QString recodeToUser(const QString &input, QUrl::ComponentFormattingOptions encoding) const;
+
+ void setQuery(const QString &query);
+
+ void addQueryItem(const QString &key, const QString &value)
+ { itemList.append(qMakePair(recodeFromUser(key), recodeFromUser(value))); }
+ int findRecodedKey(const QString &key, int from = 0) const
+ {
+ for (int i = from; i < itemList.size(); ++i)
+ if (itemList.at(i).first == key)
+ return i;
+ return itemList.size();
+ }
+ Map::const_iterator findKey(const QString &key) const
+ { return itemList.constBegin() + findRecodedKey(recodeFromUser(key)); }
+ Map::iterator findKey(const QString &key)
+ { return itemList.begin() + findRecodedKey(recodeFromUser(key)); }
+
+ // use QMap so we end up sorting the items by key
+ Map itemList;
+ QChar valueDelimiter;
+ QChar pairDelimiter;
+};
+
+template<> void QSharedDataPointer<QUrlQueryPrivate>::detach()
+{
+ if (d && d->ref.load() == 1)
+ return;
+ QUrlQueryPrivate *x = (d ? new QUrlQueryPrivate(*d)
+ : new QUrlQueryPrivate);
+ x->ref.ref();
+ if (d && !d->ref.deref())
+ delete d;
+ d = x;
+}
+
+// Here's how we do the encoding in QUrlQuery
+// The RFC says these are the delimiters:
+// gen-delims = ":" / "/" / "?" / "#" / "[" / "]" / "@"
+// sub-delims = "!" / "$" / "&" / "'" / "(" / ")"
+// / "*" / "+" / "," / ";" / "="
+// And the definition of query is:
+// query = *( pchar / "/" / "?" )
+// pchar = unreserved / pct-encoded / sub-delims / ":" / "@"
+//
+// The strict definition of query says that it can have unencoded any
+// unreserved, sub-delim, ":", "@", "/" and "?". Or, by exclusion, excluded
+// delimiters are "#", "[" and "]" -- if those are present, they must be
+// percent-encoded.
+//
+// The internal storage in the Map is equivalent to PrettyDecoded. That means
+// the getter methods, when called with the default encoding value, will not
+// have to recode anything (except for toString()).
+//
+// The "+" sub-delimiter is always left untouched. We never encode "+" to "%2B"
+// nor do we decode "%2B" to "+", no matter what the user asks.
+//
+// The rest of the delimiters are kept in their decoded forms and that's
+// considered non-ambiguous. That includes the pair and value delimiters
+// themselves.
+//
+// But when recreating the query string, in toString(), we must take care of
+// the special delimiters: the pair and value delimiters and those that the
+// definition says must be encoded ("#" / "[" / "]")
+
+#define decode(x) ushort(x)
+#define leave(x) ushort(0x100 | (x))
+#define encode(x) ushort(0x200 | (x))
+static const ushort prettyDecodedActions[] = { leave('+'), 0 };
+
+inline QString QUrlQueryPrivate::recodeFromUser(const QString &input) const
+{
+ // note: duplicated in setQuery()
+ QString output;
+ if (qt_urlRecode(output, input.constData(), input.constData() + input.length(),
+ QUrl::DecodeUnicode | QUrl::DecodeAllDelimiters | QUrl::DecodeSpaces,
+ prettyDecodedActions))
+ return output;
+ return input;
+}
+
+inline bool idempotentRecodeToUser(QUrl::ComponentFormattingOptions encoding)
+{
+ return encoding == QUrl::PrettyDecoded || encoding == (QUrl::PrettyDecoded | QUrl::DecodeAllDelimiters);
+}
+
+inline QString QUrlQueryPrivate::recodeToUser(const QString &input, QUrl::ComponentFormattingOptions encoding) const
+{
+ // our internal formats are stored in "PrettyDecoded" form
+ // and there are no ambiguous characters
+ if (idempotentRecodeToUser(encoding))
+ return input;
+
+ bool decodeUnambiguous = encoding & QUrl::DecodeUnambiguousDelimiters;
+ encoding &= ~QUrl::DecodeAllDelimiters;
+
+ if (decodeUnambiguous) {
+ QString output;
+ if (qt_urlRecode(output, input.constData(), input.constData() + input.length(),
+ encoding | QUrl::DecodeAllDelimiters, prettyDecodedActions))
+ return output;
+ return input;
+ }
+
+ // re-encode the gen-delims that the RFC says must be encoded the
+ // query delimiter pair; the non-delimiters will get encoded too
+ ushort actions[] = { encode(pairDelimiter.unicode()), encode(valueDelimiter.unicode()),
+ encode('#'), encode('['), encode(']'), 0 };
+ QString output;
+ if (qt_urlRecode(output, input.constData(), input.constData() + input.length(), encoding, actions))
+ return output;
+ return input;
+}
+
+void QUrlQueryPrivate::setQuery(const QString &query)
+{
+ itemList.clear();
+ const QChar *pos = query.constData();
+ const QChar *const end = pos + query.size();
+ while (pos != end) {
+ const QChar *begin = pos;
+ const QChar *delimiter = 0;
+ while (pos != end) {
+ // scan for the component parts of this pair
+ if (!delimiter && pos->unicode() == valueDelimiter)
+ delimiter = pos;
+ if (pos->unicode() == pairDelimiter)
+ break;
+ ++pos;
+ }
+ if (!delimiter)
+ delimiter = pos;
+
+ // pos is the end of this pair (the end of the string or the pair delimiter)
+ // delimiter points to the value delimiter or to the end of this pair
+
+ QString key;
+ if (!qt_urlRecode(key, begin, delimiter,
+ QUrl::DecodeUnicode | QUrl::DecodeAllDelimiters | QUrl::DecodeSpaces,
+ prettyDecodedActions))
+ key = QString(begin, delimiter - begin);
+
+ if (delimiter == pos) {
+ // the value delimiter wasn't found, store a null value
+ itemList.append(qMakePair(key, QString()));
+ } else if (delimiter + 1 == pos) {
+ // if the delimiter was found but the value is empty, store empty-but-not-null
+ itemList.append(qMakePair(key, QString(0, Qt::Uninitialized)));
+ } else {
+ QString value;
+ if (!qt_urlRecode(value, delimiter + 1, pos,
+ QUrl::DecodeUnicode | QUrl::DecodeAllDelimiters | QUrl::DecodeSpaces,
+ prettyDecodedActions))
+ value = QString(delimiter + 1, pos - delimiter - 1);
+ itemList.append(qMakePair(key, value));
+ }
+
+ if (pos != end)
+ ++pos;
+ }
+}
+
+// allow QUrlQueryPrivate to detach from null
+template <> inline QUrlQueryPrivate *
+QSharedDataPointer<QUrlQueryPrivate>::clone()
+{
+ return d ? new QUrlQueryPrivate(*d) : new QUrlQueryPrivate;
+}
+
+/*!
+ Constructs an empty QUrlQuery object. A query can be set afterwards by
+ calling setQuery() or items can be added by using addQueryItem().
+
+ \sa setQuery(), addQueryItem()
+*/
+QUrlQuery::QUrlQuery()
+ : d(0)
+{
+}
+
+/*!
+ Constructs a QUrlQuery object and parses the \a queryString query string,
+ using the default query delimiters. To parse a query string using other
+ delimiters, you should first set them using setQueryDelimiters() and then
+ set the query with setQuery().
+*/
+QUrlQuery::QUrlQuery(const QString &queryString)
+ : d(queryString.isEmpty() ? 0 : new QUrlQueryPrivate(queryString))
+{
+}
+
+/*!
+ Constructs a QUrlQuery object and parses the query string found in the \a
+ url URL, using the default query delimiters. To parse a query string using
+ other delimiters, you should first set them using setQueryDelimiters() and
+ then set the query with setQuery().
+
+ \sa QUrl::query()
+*/
+QUrlQuery::QUrlQuery(const QUrl &url)
+ : d(0)
+{
+ // use internals to avoid unnecessary recoding
+ // ### FIXME: actually do it
+ if (url.hasQuery())
+ d = new QUrlQueryPrivate(QString::fromUtf8(url.encodedQuery()));
+}
+
+/*!
+ Copies the contents of the \a other QUrlQuery object, including the query
+ delimiters.
+*/
+QUrlQuery::QUrlQuery(const QUrlQuery &other)
+ : d(other.d)
+{
+}
+
+/*!
+ Copies the contents of the \a other QUrlQuery object, including the query
+ delimiters.
+*/
+QUrlQuery &QUrlQuery::operator =(const QUrlQuery &other)
+{
+ d = other.d;
+ return *this;
+}
+
+/*!
+ Destroys this QUrlQuery object.
+*/
+QUrlQuery::~QUrlQuery()
+{
+ // d auto-deletes
+}
+
+/*!
+ Returns true if this object and the \a other object contain the same
+ contents, in the same order, and use the same query delimiters.
+*/
+bool QUrlQuery::operator ==(const QUrlQuery &other) const
+{
+ if (d == other.d)
+ return true;
+ if (d && other.d)
+ return d->valueDelimiter == other.d->valueDelimiter &&
+ d->pairDelimiter == other.d->pairDelimiter &&
+ d->itemList == other.d->itemList;
+ return false;
+}
+
+/*!
+ Returns true if this QUrlQUery object contains no key-value pairs, such as
+ after being default-constructed or after parsing an empty query string.
+
+ \sa setQuery(), clear()
+*/
+bool QUrlQuery::isEmpty() const
+{
+ return d ? d->itemList.isEmpty() : true;
+}
+
+/*!
+ \internal
+*/
+bool QUrlQuery::isDetached() const
+{
+ return d && d->ref.load() == 1;
+}
+
+/*!
+ Clears this QUrlQuery object by removing all of the key-value pairs
+ currently stored. If the query delimiters have been changed, this function
+ will leave them with their changed values.
+
+ \sa isEmpty(), setQueryDelimiters()
+*/
+void QUrlQuery::clear()
+{
+ if (d.constData())
+ d->itemList.clear();
+}
+
+/*!
+ Parses the query string in \a queryString and sets the internal items to
+ the values found there. If any delimiters have been specified with
+ setQueryDelimiters(), this function will use them instead of the default
+ delimiters to parse the string.
+*/
+void QUrlQuery::setQuery(const QString &queryString)
+{
+ d->setQuery(queryString);
+}
+
+static void recodeAndAppend(QString &to, const QString &input,
+ QUrl::ComponentFormattingOptions encoding, const ushort *tableModifications)
+{
+ if (!qt_urlRecode(to, input.constData(), input.constData() + input.length(), encoding, tableModifications))
+ to += input;
+}
+
+/*!
+ Returns the reconstructed query string, formed from the key-value pairs
+ currently stored in this QUrlQuery object and separated by the query
+ delimiters chosen for this object. The keys and values are encoded using
+ the options given by the \a encoding paramter.
+
+ For this function, the only ambiguous delimiter is the hash ("#"), as in
+ URLs it is used to separate the query string from the fragment that may
+ follow.
+
+ The order of the key-value pairs in the returned string is exactly the same
+ as in the original query.
+
+ \sa setQuery(), QUrl::setQuery(), QUrl::fragment(), \l{#Encoding}{Encoding}
+*/
+QString QUrlQuery::query(QUrl::ComponentFormattingOptions encoding) const
+{
+ if (!d)
+ return QString();
+
+ // unlike the component encoding, for the whole query we need to modify a little:
+ // - the "#" character is ambiguous, so we decode it only in DecodeAllDelimiters mode
+ // - the query delimiter pair must always be encoded
+ // - the "[" and "]" sub-delimiters and the non-delimiters very on DecodeUnambiguousDelimiters
+ // so:
+ // - full encoding: encode the non-delimiters, the pair, "#", "[" and "]"
+ // - pretty decode: decode the non-delimiters, "[" and "]"; encode the pair and "#"
+ // - decode all: decode the non-delimiters, "[", "]", "#"; encode the pair
+
+ // start with what's always encoded
+ ushort tableActions[7] = {
+ leave('+'), // 0
+ encode(d->pairDelimiter.unicode()), // 1
+ encode(d->valueDelimiter.unicode()), // 2
+ };
+ if ((encoding & QUrl::DecodeAllDelimiters) == QUrl::DecodeAllDelimiters) {
+ // full decoding: we only encode the characters above
+ tableActions[3] = 0;
+ encoding |= QUrl::DecodeAllDelimiters;
+ } else {
+ tableActions[3] = encode('#');
+ if (encoding & QUrl::DecodeUnambiguousDelimiters) {
+ tableActions[4] = 0;
+ encoding |= QUrl::DecodeAllDelimiters;
+ } else {
+ tableActions[4] = encode('[');
+ tableActions[5] = encode(']');
+ tableActions[6] = 0;
+ encoding &= ~QUrl::DecodeAllDelimiters;
+ }
+ }
+
+ QString result;
+ Map::const_iterator it = d->itemList.constBegin();
+ Map::const_iterator end = d->itemList.constEnd();
+
+ {
+ int size = 0;
+ for ( ; it != end; ++it)
+ size += it->first.length() + 1 + it->second.length() + 1;
+ result.reserve(size + size / 4);
+ }
+
+ for (it = d->itemList.constBegin(); it != end; ++it) {
+ if (!result.isEmpty())
+ result += QChar(d->pairDelimiter);
+ recodeAndAppend(result, it->first, encoding, tableActions);
+ if (!it->second.isNull()) {
+ result += QChar(d->valueDelimiter);
+ recodeAndAppend(result, it->second, encoding, tableActions);
+ }
+ }
+ return result;
+}
+
+/*!
+ Sets the characters used for delimiting between keys and values,
+ and between key-value pairs in the URL's query string. The default
+ value delimiter is '=' and the default pair delimiter is '&'.
+
+ \img qurl-querystring.png
+
+ \a valueDelimiter will be used for separating keys from values,
+ and \a pairDelimiter will be used to separate key-value pairs.
+ Any occurrences of these delimiting characters in the encoded
+ representation of the keys and values of the query string are
+ percent encoded when returned in query().
+
+ If \a valueDelimiter is set to '(' and \a pairDelimiter is ')',
+ the above query string would instead be represented like this:
+
+ \snippet doc/src/snippets/code/src_corelib_io_qurl.cpp 4
+
+ \note Non-standard delimiters should be chosen from among what RFC 3986 calls
+ "sub-delimiters". They are:
+
+ \code
+ sub-delims = "!" / "$" / "&" / "'" / "(" / ")"
+ / "*" / "+" / "," / ";" / "="
+ \endcode
+
+ Use of other characters is not supported and may result in unexpected
+ behaviour. This method does not verify that you passed a valid delimiter.
+
+ \sa queryValueDelimiter(), queryPairDelimiter()
+*/
+void QUrlQuery::setQueryDelimiters(QChar valueDelimiter, QChar pairDelimiter)
+{
+ d->valueDelimiter = valueDelimiter.unicode();
+ d->pairDelimiter = pairDelimiter.unicode();
+}
+
+/*!
+ Returns the character used to delimit between keys and values when
+ reconstructing the query string in query() or when parsing in setQuery().
+
+ \sa setQueryDelimiters(), queryPairDelimiter()
+*/
+QChar QUrlQuery::queryValueDelimiter() const
+{
+ return d ? d->valueDelimiter : defaultQueryValueDelimiter();
+}
+
+/*!
+ Returns the character used to delimit between keys-value pairs when
+ reconstructing the query string in query() or when parsing in setQuery().
+
+ \sa setQueryDelimiters(), queryValueDelimiter()
+*/
+QChar QUrlQuery::queryPairDelimiter() const
+{
+ return d ? d->pairDelimiter : defaultQueryPairDelimiter();
+}
+
+/*!
+ Sets the items in this QUrlQuery object to \a query. The order of the
+ elements in \a query is preserved.
+
+ \note This method does not treat spaces (ASCII 0x20) and plus ("+") signs
+ as the same, like HTML forms do. If you need spaces to be represented as
+ plus signs, use actual plus signs.
+
+ \sa queryItems(), isEmpty()
+*/
+void QUrlQuery::setQueryItems(const QList<QPair<QString, QString> > &query)
+{
+ clear();
+ if (query.isEmpty())
+ return;
+
+ QUrlQueryPrivate *dd = d;
+ QList<QPair<QString, QString> >::const_iterator it = query.constBegin(),
+ end = query.constEnd();
+ for ( ; it != end; ++it)
+ dd->addQueryItem(it->first, it->second);
+}
+
+/*!
+ Returns the query string of the URL, as a map of keys and values, using the
+ options specified in \a encoding to encode the items. The order of the
+ elements is the same as the one found in the query string or set with
+ setQueryItems().
+
+ \sa setQueryItems(), \l{#Encoding}{Encoding}
+*/
+QList<QPair<QString, QString> > QUrlQuery::queryItems(QUrl::ComponentFormattingOptions encoding) const
+{
+ if (!d)
+ return QList<QPair<QString, QString> >();
+ if (idempotentRecodeToUser(encoding))
+ return d->itemList;
+
+ QList<QPair<QString, QString> > result;
+ Map::const_iterator it = d->itemList.constBegin();
+ Map::const_iterator end = d->itemList.constEnd();
+ for ( ; it != end; ++it)
+ result << qMakePair(d->recodeToUser(it->first, encoding),
+ d->recodeToUser(it->second, encoding));
+ return result;
+}
+
+/*!
+ Returns true if there is a query string pair whose key is equal
+ to \a key from the URL.
+
+ \sa addQueryItem(), queryItemValue()
+*/
+bool QUrlQuery::hasQueryItem(const QString &key) const
+{
+ if (!d)
+ return false;
+ return d->findKey(key) != d->itemList.constEnd();
+}
+
+/*!
+ Appends the pair \a key = \a value to the end of the query string of the
+ URL. This method does not overwrite existing items that might exist with
+ the same key.
+
+ \note This method does not treat spaces (ASCII 0x20) and plus ("+") signs
+ as the same, like HTML forms do. If you need spaces to be represented as
+ plus signs, use actual plus signs.
+
+ \sa hasQueryItem(), queryItemValue()
+*/
+void QUrlQuery::addQueryItem(const QString &key, const QString &value)
+{
+ d->addQueryItem(key, value);
+}
+
+/*!
+ Returns the query value associated with key \a key from the URL, using the
+ options specified in \a encoding to encode the return value. If the key \a
+ key is not found, this function returns an empty string. If you need to
+ distinguish between an empty value and a non-existent key, you should check
+ for the key's presence first using hasQueryItem().
+
+ If the key \a key is multiply defined, this function will return the first
+ one found, in the order they were present in the query string or added
+ using addQueryItem().
+
+ \sa addQueryItem(), allQueryItemValues(), \l{#Encoding}{Encoding}
+*/
+QString QUrlQuery::queryItemValue(const QString &key, QUrl::ComponentFormattingOptions encoding) const
+{
+ QString result;
+ if (d) {
+ Map::const_iterator it = d->findKey(key);
+ if (it != d->itemList.constEnd())
+ result = d->recodeToUser(it->second, encoding);
+ }
+ return result;
+}
+
+/*!
+ Returns the a list of query string values whose key is equal to \a key from
+ the URL, using the options specified in \a encoding to encode the return
+ value. If the key \a key is not found, this function returns an empty list.
+
+ \sa queryItemValue(), addQueryItem()
+*/
+QStringList QUrlQuery::allQueryItemValues(const QString &key, QUrl::ComponentFormattingOptions encoding) const
+{
+ QStringList result;
+ if (d) {
+ QString encodedKey = d->recodeFromUser(key);
+ int idx = d->findRecodedKey(encodedKey);
+ while (idx < d->itemList.size()) {
+ result << d->recodeToUser(d->itemList.at(idx).second, encoding);
+ idx = d->findRecodedKey(encodedKey, idx + 1);
+ }
+ }
+ return result;
+}
+
+/*!
+ Removes the query string pair whose key is equal to \a key from the URL. If
+ there are multiple items with a key equal to \a key, it removes the first
+ item in the order they were present in the query string or added with
+ addQueryItem().
+
+ \sa removeAllQueryItems()
+*/
+void QUrlQuery::removeQueryItem(const QString &key)
+{
+ if (d) {
+ Map::iterator it = d->findKey(key);
+ if (it != d->itemList.end())
+ d->itemList.erase(it);
+ }
+}
+
+/*!
+ Removes all the query string pairs whose key is equal to \a key
+ from the URL.
+
+ \sa removeQueryItem()
+*/
+void QUrlQuery::removeAllQueryItems(const QString &key)
+{
+ if (d.constData()) {
+ QString encodedKey = d->recodeFromUser(key);
+ Map::iterator it = d->itemList.begin();
+ while (it != d->itemList.end()) {
+ if (it->first == encodedKey)
+ it = d->itemList.erase(it);
+ else
+ ++it;
+ }
+ }
+}
+
+/*!
+ \fn QChar QUrlQuery::defaultQueryValueDelimiter()
+ Returns the default character for separating keys from values in the query,
+ an equal sign ("=").
+
+ \sa setQueryDelimiters(), queryValueDelimiter(), defaultQueryPairDelimiter()
+*/
+
+/*!
+ \fn QChar QUrlQuery::defaultQueryPairDelimiter()
+ Returns the default character for separating keys-value pairs from each
+ other, an ampersand ("&").
+
+ \sa setQueryDelimiters(), queryPairDelimiter(), defaultQueryValueDelimiter()
+*/
+
+QT_END_NAMESPACE