/* * Copyright (C) 2012 Felix Geyer * Copyright (C) 2017 KeePassXC Team * * 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 or (at your option) * version 3 of the License. * * 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 program. If not, see . */ #include "EntryAttributes.h" const QString EntryAttributes::TitleKey = "Title"; const QString EntryAttributes::UserNameKey = "UserName"; const QString EntryAttributes::PasswordKey = "Password"; const QString EntryAttributes::URLKey = "URL"; const QString EntryAttributes::NotesKey = "Notes"; const QStringList EntryAttributes::DefaultAttributes(QStringList() << TitleKey << UserNameKey << PasswordKey << URLKey << NotesKey); const QString EntryAttributes::RememberCmdExecAttr = "_EXEC_CMD"; EntryAttributes::EntryAttributes(QObject* parent) : QObject(parent) , m_referenceRegExp("\\{REF:([TUPAN])@I:([^}]+)\\}", Qt::CaseInsensitive, QRegExp::RegExp2) { clear(); } QList EntryAttributes::keys() const { return m_attributes.keys(); } bool EntryAttributes::hasKey(const QString& key) const { return m_attributes.keys().contains(key); } QList EntryAttributes::customKeys() { QList customKeys; const QList keyList = keys(); for (const QString& key : keyList) { if (!isDefaultAttribute(key)) { customKeys.append(key); } } return customKeys; } QString EntryAttributes::value(const QString& key) const { return m_attributes.value(key); } bool EntryAttributes::contains(const QString &key) const { return m_attributes.contains(key); } bool EntryAttributes::isProtected(const QString& key) const { return m_protectedAttributes.contains(key); } bool EntryAttributes::isReference(const QString& key) const { if (!m_attributes.contains(key)) { Q_ASSERT(false); return false; } QString data = value(key); if (m_referenceRegExp.indexIn(data) != -1) { return true; } return false; } QRegExp* EntryAttributes::referenceRegExp() { return &m_referenceRegExp; } void EntryAttributes::set(const QString& key, const QString& value, bool protect) { bool emitModified = false; bool addAttribute = !m_attributes.contains(key); bool changeValue = !addAttribute && (m_attributes.value(key) != value); bool defaultAttribute = isDefaultAttribute(key); if (addAttribute && !defaultAttribute) { emit aboutToBeAdded(key); } if (addAttribute || changeValue) { m_attributes.insert(key, value); emitModified = true; } if (protect) { if (!m_protectedAttributes.contains(key)) { emitModified = true; } m_protectedAttributes.insert(key); } else if (m_protectedAttributes.remove(key)) { emitModified = true; } if (emitModified) { emit modified(); } if (defaultAttribute && changeValue) { emit defaultKeyModified(); } else if (addAttribute) { emit added(key); } else if (emitModified) { emit customKeyModified(key); } } void EntryAttributes::remove(const QString& key) { Q_ASSERT(!isDefaultAttribute(key)); if (!m_attributes.contains(key)) { Q_ASSERT(false); return; } emit aboutToBeRemoved(key); m_attributes.remove(key); m_protectedAttributes.remove(key); emit removed(key); emit modified(); } void EntryAttributes::rename(const QString& oldKey, const QString& newKey) { Q_ASSERT(!isDefaultAttribute(oldKey)); Q_ASSERT(!isDefaultAttribute(newKey)); if (!m_attributes.contains(oldKey)) { Q_ASSERT(false); return; } if (m_attributes.contains(newKey)) { Q_ASSERT(false); return; } QString data = value(oldKey); bool protect = isProtected(oldKey); emit aboutToRename(oldKey, newKey); m_attributes.remove(oldKey); m_attributes.insert(newKey, data); if (protect) { m_protectedAttributes.remove(oldKey); m_protectedAttributes.insert(newKey); } emit modified(); emit renamed(oldKey, newKey); } void EntryAttributes::copyCustomKeysFrom(const EntryAttributes* other) { if (!areCustomKeysDifferent(other)) { return; } emit aboutToBeReset(); // remove all non-default keys const QList keyList = keys(); for (const QString& key : keyList) { if (!isDefaultAttribute(key)) { m_attributes.remove(key); m_protectedAttributes.remove(key); } } const QList otherKeyList = other->keys(); for (const QString& key : otherKeyList) { if (!isDefaultAttribute(key)) { m_attributes.insert(key, other->value(key)); if (other->isProtected(key)) { m_protectedAttributes.insert(key); } } } emit reset(); emit modified(); } bool EntryAttributes::areCustomKeysDifferent(const EntryAttributes* other) { // check if they are equal ignoring the order of the keys if (keys().toSet() != other->keys().toSet()) { return true; } const QList keyList = keys(); for (const QString& key : keyList) { if (isDefaultAttribute(key)) { continue; } if (isProtected(key) != other->isProtected(key) || value(key) != other->value(key)) { return true; } } return false; } void EntryAttributes::copyDataFrom(const EntryAttributes* other) { if (*this != *other) { emit aboutToBeReset(); m_attributes = other->m_attributes; m_protectedAttributes = other->m_protectedAttributes; emit reset(); emit modified(); } } bool EntryAttributes::operator==(const EntryAttributes& other) const { return (m_attributes == other.m_attributes && m_protectedAttributes == other.m_protectedAttributes); } bool EntryAttributes::operator!=(const EntryAttributes& other) const { return (m_attributes != other.m_attributes || m_protectedAttributes != other.m_protectedAttributes); } void EntryAttributes::clear() { emit aboutToBeReset(); m_attributes.clear(); m_protectedAttributes.clear(); for (const QString& key : DefaultAttributes) { m_attributes.insert(key, ""); } emit reset(); emit modified(); } int EntryAttributes::attributesSize() { int size = 0; QMapIterator i(m_attributes); while (i.hasNext()) { i.next(); size += i.value().toUtf8().size(); } return size; } bool EntryAttributes::isDefaultAttribute(const QString& key) { return DefaultAttributes.contains(key); }