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) {
187 QRegularExpression regexp(
"([A-Za-z]\\w+)/([A-Za-z]\\w+):([A-Za-z]\\w+)");
188 regexp.setPatternOptions(QRegularExpression::UseUnicodePropertiesOption);
190 QRegularExpressionMatch match;
191 if (tagString.contains(regexp, &match)) {
192 structName = match.captured(1);
193 tagName = match.captured(3);
203 QRegularExpression regexp2(
"([A-Za-z]\\w+)\\[(\\d+)\\]/([A-Za-z]\\w+):([A-Za-z]\\w+)");
204 regexp2.setPatternOptions(QRegularExpression::UseUnicodePropertiesOption);
206 QRegularExpressionMatch match2;
207 if (tagString.contains(regexp2, &match2)) {
208 structName = match2.captured(1);
209 arrayIndex = match2.captured(2).toInt() - 1;
210 tagName = match2.captured(4);
225 warnKrita <<
"WARNING: Unsupported tag. We do not yet support nested tags. The tag will be dropped!";
226 warnKrita <<
" Failing tag:" << tagString;
232 ioDevice->open(QIODevice::ReadOnly);
234 std::string xmpPacket_;
235 QByteArray arr = ioDevice->readAll();
236 xmpPacket_.assign(arr.data(), arr.length());
239 Exiv2::XmpData xmpData_;
240 if (Exiv2::XmpParser::decode(xmpData_, xmpPacket_) != 0) {
244 QMap<const KisMetaData::Schema *, QMap<QString, QMap<QString, KisMetaData::Value>>> structures;
245 QMap<const KisMetaData::Schema *, QMap<QString, QVector<QMap<QString, KisMetaData::Value>>>> arraysOfStructures;
246 for (Exiv2::XmpData::iterator it = xmpData_.begin(); it != xmpData_.end(); ++it) {
247 dbgMetaData <<
"Start iteration" << it->key().c_str();
249 Exiv2::XmpKey key(it->key());
250 dbgMetaData << key.groupName().c_str() <<
" " << key.tagName().c_str() <<
" " << key.ns().c_str();
251 if ((key.groupName() ==
"exif" || key.groupName() ==
"tiff")
252 && key.tagName() ==
"NativeDigest") {
266#if EXIV2_TEST_VERSION(0,28,0)
267 const Exiv2::Value::UniquePtr
value = it->getValue();
269 const Exiv2::Value::AutoPtr
value = it->getValue();
276 if (!
parseTagName(key.tagName().c_str(), structName, arrayIndex, tagName, &typeInfo, schema))
279 bool isStructureEntry = !structName.isEmpty() && arrayIndex == -1;
280 bool isStructureInArrayEntry = !structName.isEmpty() && arrayIndex != -1;
281 Q_ASSERT(isStructureEntry != isStructureInArrayEntry || !isStructureEntry);
284 bool ignoreValue =
false;
286 if (
value->typeId() == Exiv2::xmpBag ||
value->typeId() == Exiv2::xmpSeq
287 ||
value->typeId() == Exiv2::xmpAlt) {
293 if (embeddedTypeInfo) {
294 parser = embeddedTypeInfo->
parser();
296 const Exiv2::XmpArrayValue *xav =
dynamic_cast<const Exiv2::XmpArrayValue *
>(
value.get());
299#if EXIV2_TEST_VERSION(0,28,0)
300 for (
size_t i = 0; i < xav->count(); ++i) {
302 for (
int i = 0; i < xav->count(); ++i) {
304 QString
value = QString::fromStdString(xav->toString(i));
308 dbgImage <<
"No parser " << tagName;
313 switch (xav->xmpArrayType()) {
314 case Exiv2::XmpValue::xaNone:
315 warnKrita <<
"KisXMPIO: Unsupported array";
317 case Exiv2::XmpValue::xaAlt:
320 case Exiv2::XmpValue::xaBag:
323 case Exiv2::XmpValue::xaSeq:
328 }
else if (
value->typeId() == Exiv2::langAlt) {
329 const Exiv2::LangAltValue *xav =
dynamic_cast<const Exiv2::LangAltValue *
>(
value.get());
333 for (std::map<std::string, std::string>::const_iterator it = xav->value_.begin();
334 it != xav->value_.end();
342 QString valTxt =
value->toString().c_str();
343 if (typeInfo && typeInfo->
parser()) {
349 if (valTxt ==
"type=\"Struct\"") {
357 if (isStructureEntry) {
358 structures[schema][structName][tagName] =
v;
359 }
else if (isStructureInArrayEntry) {
360 if (arraysOfStructures[schema][structName].size() <= arrayIndex) {
361 arraysOfStructures[schema][structName].resize(arrayIndex + 1);
364 if (!arraysOfStructures[schema][structName][arrayIndex].contains(tagName)) {
365 arraysOfStructures[schema][structName][arrayIndex][tagName] =
v;
367 warnKrita <<
"WARNING: trying to overwrite tag" << tagName <<
"in" << structName << arrayIndex;
373 dbgMetaData <<
"Ignoring value for " << tagName <<
" " <<
v;
379 for (QMap<
const KisMetaData::Schema *, QMap<QString, QMap<QString, KisMetaData::Value>>>::iterator it =
381 it != structures.end();
384 for (QMap<QString, QMap<QString, KisMetaData::Value>>::iterator it2 = it.value().begin();
385 it2 != it.value().end();
391 arraysOfStructures.begin();
392 it != arraysOfStructures.end();
395 for (QMap<QString,
QVector<QMap<QString, KisMetaData::Value>>>::iterator it2 = it.value().begin();
396 it2 != it.value().end();
399 QString entryName = it2.key();
417 if (
value.isArray()) {
424 for (
int i = 0; i < it2.value().size(); ++i) {
425 valueList.append(it2.value()[i]);
bool parseTagName(const QString &tagString, QString &structName, int &arrayIndex, QString &tagName, const KisMetaData::TypeInfo **typeInfo, const KisMetaData::Schema *schema)