72 Exiv2::XmpData xmpData_;
83 QMap<QString, KisMetaData::Value> structure =
value.asStructure();
88 if (!structureSchema) {
89 dbgMetaData <<
"Unknown schema for " << entry.name();
90 structureSchema = entry.schema();
92 Q_ASSERT(structureSchema);
93 saveStructure(xmpData_, entry.name(), prefix, structure, structureSchema);
95 Exiv2::XmpKey key(prefix, entry.name().toLatin1().constData());
102 Exiv2::XmpTextValue tv;
105 tv.setXmpArrayType(Exiv2::XmpValue::xaSeq);
108 tv.setXmpArrayType(Exiv2::XmpValue::xaBag);
111 tv.setXmpArrayType(Exiv2::XmpValue::xaAlt);
117 xmpData_.add(key, &tv);
120 if (structureTypeInfo) {
123 if (!structureSchema) {
124 dbgMetaData <<
"Unknown schema for " << entry.name();
125 structureSchema = entry.schema();
127 Q_ASSERT(structureSchema);
129 for (
int idx = 0; idx < array.size(); ++idx) {
130 saveStructure(xmpData_,
131 QString(
"%1[%2]").arg(entry.name()).arg(idx + 1),
133 array[idx].asStructure(),
140 xmpData_.add(key,
v);
147 std::string xmpPacket_;
149 Exiv2::XmpParser::encode(xmpPacket_, xmpData_);
150 }
catch (std::exception &e) {
151 warnMetaData <<
"Couldn't encode the data, error =" << e.what();
155 ioDevice->open(QIODevice::WriteOnly);
157 xmpPacket_ =
"http://ns.adobe.com/xap/1.0/\0" + xmpPacket_;
159 ioDevice->write(xmpPacket_.c_str(), xmpPacket_.length());
173 int numSubNames = tagString.count(
'/') + 1;
175 if (numSubNames == 1) {
182 if (numSubNames == 2) {
183 QRegExp regexp(
"([A-Za-z]\\w+)/([A-Za-z]\\w+):([A-Za-z]\\w+)");
184 if (regexp.indexIn(tagString) != -1) {
185 structName = regexp.capturedTexts()[1];
186 tagName = regexp.capturedTexts()[3];
196 QRegExp regexp2(
"([A-Za-z]\\w+)\\[(\\d+)\\]/([A-Za-z]\\w+):([A-Za-z]\\w+)");
197 if (regexp2.indexIn(tagString) != -1) {
198 structName = regexp2.capturedTexts()[1];
199 arrayIndex = regexp2.capturedTexts()[2].toInt() - 1;
200 tagName = regexp2.capturedTexts()[4];
215 warnKrita <<
"WARNING: Unsupported tag. We do not yet support nested tags. The tag will be dropped!";
216 warnKrita <<
" Failing tag:" << tagString;
222 ioDevice->open(QIODevice::ReadOnly);
224 std::string xmpPacket_;
225 QByteArray arr = ioDevice->readAll();
226 xmpPacket_.assign(arr.data(), arr.length());
229 Exiv2::XmpData xmpData_;
230 if (Exiv2::XmpParser::decode(xmpData_, xmpPacket_) != 0) {
234 QMap<const KisMetaData::Schema *, QMap<QString, QMap<QString, KisMetaData::Value>>> structures;
235 QMap<const KisMetaData::Schema *, QMap<QString, QVector<QMap<QString, KisMetaData::Value>>>> arraysOfStructures;
236 for (Exiv2::XmpData::iterator it = xmpData_.begin(); it != xmpData_.end(); ++it) {
237 dbgMetaData <<
"Start iteration" << it->key().c_str();
239 Exiv2::XmpKey key(it->key());
240 dbgMetaData << key.groupName().c_str() <<
" " << key.tagName().c_str() <<
" " << key.ns().c_str();
241 if ((key.groupName() ==
"exif" || key.groupName() ==
"tiff")
242 && key.tagName() ==
"NativeDigest") {
256#if EXIV2_TEST_VERSION(0,28,0)
257 const Exiv2::Value::UniquePtr
value = it->getValue();
259 const Exiv2::Value::AutoPtr
value = it->getValue();
266 if (!
parseTagName(key.tagName().c_str(), structName, arrayIndex, tagName, &typeInfo, schema))
269 bool isStructureEntry = !structName.isEmpty() && arrayIndex == -1;
270 bool isStructureInArrayEntry = !structName.isEmpty() && arrayIndex != -1;
271 Q_ASSERT(isStructureEntry != isStructureInArrayEntry || !isStructureEntry);
274 bool ignoreValue =
false;
276 if (
value->typeId() == Exiv2::xmpBag ||
value->typeId() == Exiv2::xmpSeq
277 ||
value->typeId() == Exiv2::xmpAlt) {
283 if (embeddedTypeInfo) {
284 parser = embeddedTypeInfo->
parser();
286 const Exiv2::XmpArrayValue *xav =
dynamic_cast<const Exiv2::XmpArrayValue *
>(
value.get());
289#if EXIV2_TEST_VERSION(0,28,0)
290 for (
size_t i = 0; i < xav->count(); ++i) {
292 for (
int i = 0; i < xav->count(); ++i) {
294 QString
value = QString::fromStdString(xav->toString(i));
298 dbgImage <<
"No parser " << tagName;
303 switch (xav->xmpArrayType()) {
304 case Exiv2::XmpValue::xaNone:
305 warnKrita <<
"KisXMPIO: Unsupported array";
307 case Exiv2::XmpValue::xaAlt:
310 case Exiv2::XmpValue::xaBag:
313 case Exiv2::XmpValue::xaSeq:
318 }
else if (
value->typeId() == Exiv2::langAlt) {
319 const Exiv2::LangAltValue *xav =
dynamic_cast<const Exiv2::LangAltValue *
>(
value.get());
323 for (std::map<std::string, std::string>::const_iterator it = xav->value_.begin();
324 it != xav->value_.end();
332 QString valTxt =
value->toString().c_str();
333 if (typeInfo && typeInfo->
parser()) {
339 if (valTxt ==
"type=\"Struct\"") {
347 if (isStructureEntry) {
348 structures[schema][structName][tagName] =
v;
349 }
else if (isStructureInArrayEntry) {
350 if (arraysOfStructures[schema][structName].size() <= arrayIndex) {
351 arraysOfStructures[schema][structName].resize(arrayIndex + 1);
354 if (!arraysOfStructures[schema][structName][arrayIndex].contains(tagName)) {
355 arraysOfStructures[schema][structName][arrayIndex][tagName] =
v;
357 warnKrita <<
"WARNING: trying to overwrite tag" << tagName <<
"in" << structName << arrayIndex;
363 dbgMetaData <<
"Ignoring value for " << tagName <<
" " <<
v;
369 for (QMap<
const KisMetaData::Schema *, QMap<QString, QMap<QString, KisMetaData::Value>>>::iterator it =
371 it != structures.end();
374 for (QMap<QString, QMap<QString, KisMetaData::Value>>::iterator it2 = it.value().begin();
375 it2 != it.value().end();
381 arraysOfStructures.begin();
382 it != arraysOfStructures.end();
385 for (QMap<QString,
QVector<QMap<QString, KisMetaData::Value>>>::iterator it2 = it.value().begin();
386 it2 != it.value().end();
389 QString entryName = it2.key();
407 if (
value.isArray()) {
414 for (
int i = 0; i < it2.value().size(); ++i) {
415 valueList.append(it2.value()[i]);
bool parseTagName(const QString &tagString, QString &structName, int &arrayIndex, QString &tagName, const KisMetaData::TypeInfo **typeInfo, const KisMetaData::Schema *schema)