summaryrefslogtreecommitdiffstats
path: root/src/versit/qvcard21writer.cpp
diff options
context:
space:
mode:
authorXizhi Zhu <xizhi.zhu@nokia.com>2011-08-10 15:08:50 +0300
committerXizhi Zhu <xizhi.zhu@nokia.com>2011-08-23 13:57:38 +0300
commitff95b1592bae300914e0e2a65d33b398f61391b1 (patch)
treeeceaee1d177252dfc2ab2dcad5f4bf20882cf01e /src/versit/qvcard21writer.cpp
Import from latest QtMobility.
Diffstat (limited to 'src/versit/qvcard21writer.cpp')
-rw-r--r--src/versit/qvcard21writer.cpp278
1 files changed, 278 insertions, 0 deletions
diff --git a/src/versit/qvcard21writer.cpp b/src/versit/qvcard21writer.cpp
new file mode 100644
index 000000000..315753834
--- /dev/null
+++ b/src/versit/qvcard21writer.cpp
@@ -0,0 +1,278 @@
+/****************************************************************************
+**
+** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies).
+** All rights reserved.
+** Contact: Nokia Corporation (qt-info@nokia.com)
+**
+** This file is part of the Qt Mobility Components.
+**
+** $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 "qvcard21writer_p.h"
+#include "versitutils_p.h"
+
+#include <QTextCodec>
+
+QTPIM_BEGIN_NAMESPACE
+
+/*! Constructs a writer. */
+QVCard21Writer::QVCard21Writer(QVersitDocument::VersitType type) : QVersitDocumentWriter(type)
+{
+}
+
+QTextEncoder* QVCard21Writer::utf8Encoder()
+{
+ static QTextEncoder* encoder = 0;
+ if (encoder == 0) {
+ encoder = QTextCodec::codecForName("UTF-8")->makeEncoder();
+ // Hack so the encoder doesn't output a byte order mark
+ encoder->fromUnicode(QString());
+ }
+ return encoder;
+}
+
+/*! Destroys a writer. */
+QVCard21Writer::~QVCard21Writer()
+{
+}
+
+/*!
+ * Encodes the \a property and writes it to the device.
+ */
+void QVCard21Writer::encodeVersitProperty(const QVersitProperty& property)
+{
+ encodeGroupsAndName(property);
+ QMultiHash<QString,QString> parameters = property.parameters();
+ QVariant variant(property.variantValue());
+
+ QString renderedValue;
+ QByteArray renderedBytes;
+
+ /* Structured values need to have their components backslash-escaped (in vCard 2.1, semicolons
+ must be escaped for compound values and commas must be escaped for list values). */
+ if (variant.type() == QVariant::StringList) {
+ QStringList values = property.variantValue().toStringList();
+ QString separator;
+ if (property.valueType() == QVersitProperty::CompoundType) {
+ separator = QLatin1String(";");
+ } else {
+ if (property.valueType() != QVersitProperty::ListType) {
+ qWarning("Variant value is a QStringList but the property's value type is neither "
+ "CompoundType or ListType");
+ }
+ // Assume it's a ListType
+ separator = QLatin1String(",");
+ }
+ QString replacement = QLatin1Char('\\') + separator;
+ QRegExp separatorRegex = QRegExp(separator);
+
+ // Check first if any of the values need to be UTF-8 encoded (if so, all of them must be
+ // UTF-8 encoded)
+ bool forceUtf8 = requiresUtf8(values);
+
+ bool first = true;
+ foreach (QString value, values) {
+ if (!(value.isEmpty() && property.valueType() == QVersitProperty::ListType)) {
+ encodeVersitValue(parameters, value, forceUtf8);
+ if (!first) {
+ renderedValue += separator;
+ }
+ renderedValue += value.replace(separatorRegex, replacement);
+ first = false;
+ }
+ }
+ } else if (variant.type() == QVariant::String) {
+ renderedValue = variant.toString();
+ encodeVersitValue(parameters, renderedValue, false);
+ } else if (variant.type() == QVariant::ByteArray) {
+ parameters.replace(QLatin1String("ENCODING"), QLatin1String("BASE64"));
+ if (mCodecIsAsciiCompatible) // optimize by not converting to unicode
+ renderedBytes = variant.toByteArray().toBase64();
+ else
+ renderedValue = QLatin1String(variant.toByteArray().toBase64().data());
+ }
+
+ // Encode parameters
+ encodeParameters(parameters);
+
+ // Encode value
+ writeString(QLatin1String(":"));
+ if (variant.canConvert<QVersitDocument>()) {
+ writeCrlf();
+ QVersitDocument embeddedDocument = variant.value<QVersitDocument>();
+ encodeVersitDocument(embeddedDocument);
+ } else if (variant.type() == QVariant::String || variant.type() == QVariant::StringList) {
+ // Some devices don't support vCard-style line folding if the property is
+ // quoted-printable-encoded. Therefore, we use QP soft linebreaks if the property is being
+ // QP-encoded, and normal vCard folding otherwise.
+ if (parameters.contains(QLatin1String("ENCODING"), QLatin1String("QUOTED-PRINTABLE")))
+ writeStringQp(renderedValue);
+ else
+ writeString(renderedValue);
+ } else if (variant.type() == QVariant::ByteArray) {
+ // One extra folding before the value and
+ // one extra line break after the value are needed in vCard 2.1
+ writeCrlf();
+ writeString(QLatin1String(" "));
+ if (renderedBytes.isEmpty())
+ writeString(renderedValue);
+ else
+ writeBytes(renderedBytes);
+ writeCrlf();
+ }
+ writeCrlf();
+}
+
+/*! Returns true if and only if the current codec is incapable of encoding any of the \a values */
+bool QVCard21Writer::requiresUtf8(const QStringList& values) {
+ foreach (const QString& value, values) {
+ if (!mCodec->canEncode(value)
+ // if codec is ASCII and there is a character > U+007F in value, encode it as UTF-8
+ || (mCodecIsAscii && containsNonAscii(value))) {
+ return true;
+ }
+ }
+ return false;
+}
+
+/*! Performs Quoted-Printable encoding and charset encoding on \a value as per vCard 2.1 spec.
+ Returns true if the value will need to be encoded with UTF-8, false if mCodec is sufficient. */
+void QVCard21Writer::encodeVersitValue(QMultiHash<QString,QString>& parameters, QString& value,
+ bool forceUtf8)
+{
+ // Add the CHARSET parameter, if necessary and encode in UTF-8 later
+ if (forceUtf8
+ || !mCodec->canEncode(value)
+ // if codec is ASCII and there is a character > U+007F in value, encode it as UTF-8
+ || (mCodecIsAscii && containsNonAscii(value))) {
+ parameters.replace(QLatin1String("CHARSET"), QLatin1String("UTF-8"));
+ value = QString::fromLatin1(utf8Encoder()->fromUnicode(value));
+ }
+
+ // Quoted-Printable encode the value and add Quoted-Printable parameter, if necessary
+ if (quotedPrintableEncode(value))
+ parameters.replace(QLatin1String("ENCODING"), QLatin1String("QUOTED-PRINTABLE"));
+}
+
+int sortIndexOfTypeValue(const QString& type) {
+ if ( type == QLatin1String("CELL")
+ || type == QLatin1String("FAX")) {
+ return 0;
+ } else if (type == QLatin1String("HOME")
+ || type == QLatin1String("WORK")) {
+ return 1;
+ } else {
+ return 2;
+ }
+}
+
+bool typeValueLessThan(const QString& a, const QString& b) {
+ return sortIndexOfTypeValue(a) < sortIndexOfTypeValue(b);
+}
+
+/*! Ensure CELL and FAX are at the front because they are "more important" and some vCard
+ parsers may ignore everything after the first TYPE */
+void sortTypeValues(QStringList* values)
+{
+ qSort(values->begin(), values->end(), typeValueLessThan);
+}
+
+/*!
+ * Encodes the \a parameters and writes it to the device.
+ */
+void QVCard21Writer::encodeParameters(const QMultiHash<QString,QString>& parameters)
+{
+ QList<QString> names = parameters.uniqueKeys();
+ foreach (const QString& name, names) {
+ QStringList values = parameters.values(name);
+ if (name == QLatin1String("TYPE")) {
+ // TYPE parameters should be sorted
+ sortTypeValues(&values);
+ }
+ foreach (const QString& value, values) {
+ writeString(QLatin1String(";"));
+ if (name.length() > 0 && name != QLatin1String("TYPE")) {
+ writeString(name);
+ writeString(QLatin1String("="));
+ }
+ writeString(value);
+ }
+ }
+}
+
+bool QVCard21Writer::containsNonAscii(const QString& str)
+{
+ for (int i = 0; i < str.length(); i++) {
+ if (str[i].unicode() > 127)
+ return true;
+ }
+ return false;
+}
+
+/*!
+ * Encodes special characters in \a text
+ * using Quoted-Printable encoding (RFC 1521).
+ * Returns true if at least one character was encoded.
+ */
+bool QVCard21Writer::quotedPrintableEncode(QString& text)
+{
+ bool encoded = false;
+ for (int i=0; i<text.length(); i++) {
+ QChar current = text.at(i);
+ if (shouldBeQuotedPrintableEncoded(current)) {
+ QString encodedStr(QString::fromAscii("=%1").
+ arg(current.unicode(), 2, 16, QLatin1Char('0')).toUpper());
+ text.replace(i, 1, encodedStr);
+ i += 2;
+ encoded = true;
+ }
+ }
+ return encoded;
+}
+
+
+/*!
+ * Checks whether the \a chr should be Quoted-Printable encoded (RFC 1521).
+ */
+bool QVCard21Writer::shouldBeQuotedPrintableEncoded(QChar chr)
+{
+ int c = chr.unicode();
+ return (c < 32 ||
+ c == '!' || c == '"' || c == '#' || c == '$' ||
+ c == '=' || c == '@' || c == '[' || c == '\\' ||
+ c == ']' || c == '^' || c == '`' ||
+ (c > 122 && c < 256));
+}
+
+QTPIM_END_NAMESPACE