/* * Copyright (C) 2010 Felix Geyer * * 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 "Group.h" #include "core/Config.h" #include "core/DatabaseIcons.h" #include "core/Metadata.h" #include "core/Tools.h" const int Group::DefaultIconNumber = 48; const int Group::RecycleBinIconNumber = 43; Group::Group() : m_updateTimeinfo(true) { m_data.iconNumber = DefaultIconNumber; m_data.isExpanded = true; m_data.autoTypeEnabled = Inherit; m_data.searchingEnabled = Inherit; } Group::~Group() { // Destroy entries and children manually so DeletedObjects can be added // to database. QList entries = m_entries; Q_FOREACH (Entry* entry, entries) { delete entry; } QList children = m_children; Q_FOREACH (Group* group, children) { delete group; } if (m_db && m_parent) { DeletedObject delGroup; delGroup.deletionTime = Tools::currentDateTimeUtc(); delGroup.uuid = m_uuid; m_db->addDeletedObject(delGroup); } cleanupParent(); } Group* Group::createRecycleBin() { Group* recycleBin = new Group(); recycleBin->setUuid(Uuid::random()); recycleBin->setName(tr("Recycle Bin")); recycleBin->setIcon(RecycleBinIconNumber); recycleBin->setSearchingEnabled(Group::Disable); recycleBin->setAutoTypeEnabled(Group::Disable); return recycleBin; } template inline bool Group::set(P& property, const V& value) { if (property != value) { property = value; updateTimeinfo(); Q_EMIT modified(); return true; } else { return false; } } void Group::updateTimeinfo() { if (m_updateTimeinfo) { m_data.timeInfo.setLastModificationTime(Tools::currentDateTimeUtc()); m_data.timeInfo.setLastAccessTime(Tools::currentDateTimeUtc()); } } void Group::setUpdateTimeinfo(bool value) { m_updateTimeinfo = value; } Uuid Group::uuid() const { return m_uuid; } QString Group::name() const { return m_data.name; } QString Group::notes() const { return m_data.notes; } QImage Group::icon() const { if (m_data.customIcon.isNull()) { return databaseIcons()->icon(m_data.iconNumber); } else { Q_ASSERT(m_db); if (m_db) { return m_db->metadata()->customIcon(m_data.customIcon); } else { return QImage(); } } } QPixmap Group::iconPixmap() const { if (m_data.customIcon.isNull()) { return databaseIcons()->iconPixmap(m_data.iconNumber); } else { Q_ASSERT(m_db); QPixmap pixmap; if (m_db && !QPixmapCache::find(m_pixmapCacheKey, &pixmap)) { pixmap = QPixmap::fromImage(m_db->metadata()->customIcon(m_data.customIcon)); m_pixmapCacheKey = QPixmapCache::insert(pixmap); } return pixmap; } } int Group::iconNumber() const { return m_data.iconNumber; } Uuid Group::iconUuid() const { return m_data.customIcon; } TimeInfo Group::timeInfo() const { return m_data.timeInfo; } bool Group::isExpanded() const { return m_data.isExpanded; } QString Group::defaultAutoTypeSequence() const { return m_data.defaultAutoTypeSequence; } Group::TriState Group::autoTypeEnabled() const { return m_data.autoTypeEnabled; } Group::TriState Group::searchingEnabled() const { return m_data.searchingEnabled; } Entry* Group::lastTopVisibleEntry() const { return m_lastTopVisibleEntry; } bool Group::isExpired() const { return m_data.timeInfo.expires() && m_data.timeInfo.expiryTime() < Tools::currentDateTimeUtc(); } void Group::setUuid(const Uuid& uuid) { set(m_uuid, uuid); } void Group::setName(const QString& name) { if (set(m_data.name, name)) { Q_EMIT dataChanged(this); } } void Group::setNotes(const QString& notes) { set(m_data.notes, notes); } void Group::setIcon(int iconNumber) { Q_ASSERT(iconNumber >= 0); if (m_data.iconNumber != iconNumber || !m_data.customIcon.isNull()) { m_data.iconNumber = iconNumber; m_data.customIcon = Uuid(); m_pixmapCacheKey = QPixmapCache::Key(); updateTimeinfo(); Q_EMIT modified(); Q_EMIT dataChanged(this); } } void Group::setIcon(const Uuid& uuid) { Q_ASSERT(!uuid.isNull()); if (m_data.customIcon != uuid) { m_data.customIcon = uuid; m_data.iconNumber = 0; m_pixmapCacheKey = QPixmapCache::Key(); updateTimeinfo(); Q_EMIT modified(); Q_EMIT dataChanged(this); } } void Group::setTimeInfo(const TimeInfo& timeInfo) { m_data.timeInfo = timeInfo; } void Group::setExpanded(bool expanded) { if (m_data.isExpanded != expanded) { m_data.isExpanded = expanded; updateTimeinfo(); if (config()->get("ModifiedOnExpandedStateChanges").toBool()) { Q_EMIT modified(); } } } void Group::setDefaultAutoTypeSequence(const QString& sequence) { set(m_data.defaultAutoTypeSequence, sequence); } void Group::setAutoTypeEnabled(TriState enable) { set(m_data.autoTypeEnabled, enable); } void Group::setSearchingEnabled(TriState enable) { set(m_data.searchingEnabled, enable); } void Group::setLastTopVisibleEntry(Entry* entry) { set(m_lastTopVisibleEntry, entry); } void Group::setExpires(bool value) { if (m_data.timeInfo.expires() != value) { m_data.timeInfo.setExpires(value); updateTimeinfo(); Q_EMIT modified(); } } void Group::setExpiryTime(const QDateTime& dateTime) { if (m_data.timeInfo.expiryTime() != dateTime) { m_data.timeInfo.setExpiryTime(dateTime); updateTimeinfo(); Q_EMIT modified(); } } Group* Group::parentGroup() { return m_parent; } const Group* Group::parentGroup() const { return m_parent; } void Group::setParent(Group* parent, int index) { Q_ASSERT(parent); Q_ASSERT(index >= -1 && index <= parent->children().size()); // setting a new parent for root groups is not allowed Q_ASSERT(!m_db || (m_db->rootGroup() != this)); bool moveWithinDatabase = (m_db && m_db == parent->m_db); if (index == -1) { index = parent->children().size(); if (parentGroup() == parent) { index--; } } if (m_parent == parent && parent->children().indexOf(this) == index) { return; } if (!moveWithinDatabase) { cleanupParent(); m_parent = parent; if (m_db) { recCreateDelObjects(); // copy custom icon to the new database if (!iconUuid().isNull() && parent->m_db && m_db->metadata()->containsCustomIcon(iconUuid()) && !parent->m_db->metadata()->containsCustomIcon(iconUuid())) { parent->m_db->metadata()->addCustomIcon(iconUuid(), icon()); } } if (m_db != parent->m_db) { recSetDatabase(parent->m_db); } QObject::setParent(parent); Q_EMIT aboutToAdd(this, index); Q_ASSERT(index <= parent->m_children.size()); parent->m_children.insert(index, this); } else { Q_EMIT aboutToMove(this, parent, index); m_parent->m_children.removeAll(this); m_parent = parent; QObject::setParent(parent); Q_ASSERT(index <= parent->m_children.size()); parent->m_children.insert(index, this); } if (m_updateTimeinfo) { m_data.timeInfo.setLocationChanged(Tools::currentDateTimeUtc()); } Q_EMIT modified(); if (!moveWithinDatabase) { Q_EMIT added(); } else { Q_EMIT moved(); } } void Group::setParent(Database* db) { Q_ASSERT(db); Q_ASSERT(db->rootGroup() == this); cleanupParent(); m_parent = Q_NULLPTR; recSetDatabase(db); QObject::setParent(db); } Database* Group::database() { return m_db; } const Database* Group::database() const { return m_db; } QList Group::children() { return m_children; } const QList& Group::children() const { return m_children; } QList Group::entries() { return m_entries; } const QList& Group::entries() const { return m_entries; } QList Group::entriesRecursive(bool includeHistoryItems) const { QList entryList; entryList.append(m_entries); if (includeHistoryItems) { Q_FOREACH (Entry* entry, m_entries) { entryList.append(entry->historyItems()); } } Q_FOREACH (Group* group, m_children) { entryList.append(group->entriesRecursive(includeHistoryItems)); } return entryList; } QList Group::groupsRecursive(bool includeSelf) const { QList groupList; if (includeSelf) { groupList.append(this); } Q_FOREACH (Group* group, m_children) { groupList.append(group->groupsRecursive(true)); } return groupList; } QSet Group::customIconsRecursive() const { QSet result; if (!iconUuid().isNull()) { result.insert(iconUuid()); } Q_FOREACH (Entry* entry, entriesRecursive(true)) { if (!entry->iconUuid().isNull()) { result.insert(entry->iconUuid()); } } Q_FOREACH (Group* group, m_children) { result.unite(group->customIconsRecursive()); } return result; } Group* Group::clone(Entry::CloneFlags entryFlags) const { Group* clonedGroup = new Group(); clonedGroup->setUpdateTimeinfo(false); clonedGroup->setUuid(Uuid::random()); clonedGroup->m_data = m_data; Q_FOREACH (Entry* entry, entries()) { Entry* clonedEntry = entry->clone(entryFlags); clonedEntry->setGroup(clonedGroup); } Q_FOREACH (Group* groupChild, children()) { Group* clonedGroupChild = groupChild->clone(); clonedGroupChild->setParent(clonedGroup); } clonedGroup->setUpdateTimeinfo(true); QDateTime now = Tools::currentDateTimeUtc(); clonedGroup->m_data.timeInfo.setCreationTime(now); clonedGroup->m_data.timeInfo.setLastModificationTime(now); clonedGroup->m_data.timeInfo.setLastAccessTime(now); clonedGroup->m_data.timeInfo.setLocationChanged(now); return clonedGroup; } void Group::copyDataFrom(const Group* other) { m_data = other->m_data; m_lastTopVisibleEntry = other->m_lastTopVisibleEntry; } Database* Group::exportToDb() { Q_ASSERT(database()); Database* db = new Database(); Group* clonedGroup = clone(Entry::CloneNewUuid | Entry::CloneIncludeHistory); clonedGroup->setParent(db->rootGroup()); QSet customIcons = customIconsRecursive(); db->metadata()->copyCustomIcons(customIcons, database()->metadata()); db->copyAttributesFrom(database()); return db; } void Group::addEntry(Entry* entry) { Q_ASSERT(entry); Q_ASSERT(!m_entries.contains(entry)); Q_EMIT entryAboutToAdd(entry); m_entries << entry; connect(entry, SIGNAL(dataChanged(Entry*)), SIGNAL(entryDataChanged(Entry*))); if (m_db) { connect(entry, SIGNAL(modified()), m_db, SIGNAL(modifiedImmediate())); } Q_EMIT modified(); Q_EMIT entryAdded(entry); } void Group::removeEntry(Entry* entry) { Q_ASSERT(m_entries.contains(entry)); Q_EMIT entryAboutToRemove(entry); entry->disconnect(this); if (m_db) { entry->disconnect(m_db); } m_entries.removeAll(entry); Q_EMIT modified(); Q_EMIT entryRemoved(entry); } void Group::recSetDatabase(Database* db) { if (m_db) { disconnect(SIGNAL(dataChanged(Group*)), m_db); disconnect(SIGNAL(aboutToRemove(Group*)), m_db); disconnect(SIGNAL(removed()), m_db); disconnect(SIGNAL(aboutToAdd(Group*,int)), m_db); disconnect(SIGNAL(added()), m_db); disconnect(SIGNAL(aboutToMove(Group*,Group*,int)), m_db); disconnect(SIGNAL(moved()), m_db); disconnect(SIGNAL(modified()), m_db); } Q_FOREACH (Entry* entry, m_entries) { if (m_db) { entry->disconnect(m_db); } if (db) { connect(entry, SIGNAL(modified()), db, SIGNAL(modifiedImmediate())); } } if (db) { connect(this, SIGNAL(dataChanged(Group*)), db, SIGNAL(groupDataChanged(Group*))); connect(this, SIGNAL(aboutToRemove(Group*)), db, SIGNAL(groupAboutToRemove(Group*))); connect(this, SIGNAL(removed()), db, SIGNAL(groupRemoved())); connect(this, SIGNAL(aboutToAdd(Group*,int)), db, SIGNAL(groupAboutToAdd(Group*,int))); connect(this, SIGNAL(added()), db, SIGNAL(groupAdded())); connect(this, SIGNAL(aboutToMove(Group*,Group*,int)), db, SIGNAL(groupAboutToMove(Group*,Group*,int))); connect(this, SIGNAL(moved()), db, SIGNAL(groupMoved())); connect(this, SIGNAL(modified()), db, SIGNAL(modifiedImmediate())); } m_db = db; Q_FOREACH (Group* group, m_children) { group->recSetDatabase(db); } } void Group::cleanupParent() { if (m_parent) { Q_EMIT aboutToRemove(this); m_parent->m_children.removeAll(this); Q_EMIT modified(); Q_EMIT removed(); } } void Group::recCreateDelObjects() { if (m_db) { Q_FOREACH (Entry* entry, m_entries) { m_db->addDeletedObject(entry->uuid()); } Q_FOREACH (Group* group, m_children) { group->recCreateDelObjects(); } m_db->addDeletedObject(m_uuid); } } QList Group::search(const QString& searchTerm, Qt::CaseSensitivity caseSensitivity, bool resolveInherit) { QList searchResult; bool search; if (resolveInherit) { search = resolveSearchingEnabled(); } else if (searchingEnabled() == Disable) { search = false; } else { search = true; } if (search) { Q_FOREACH (Entry* entry, m_entries) { if (entry->match(searchTerm, caseSensitivity)) { searchResult.append(entry); } } Q_FOREACH (Group* group, m_children) { searchResult.append(group->search(searchTerm, caseSensitivity, false)); } } return searchResult; } bool Group::resolveSearchingEnabled() const { switch (m_data.searchingEnabled) { case Inherit: if (!m_parent) { return true; } else { return m_parent->resolveSearchingEnabled(); } case Enable: return true; case Disable: return false; default: Q_ASSERT(false); return false; } } bool Group::resolveAutoTypeEnabled() const { switch (m_data.autoTypeEnabled) { case Inherit: if (!m_parent) { return true; } else { return m_parent->resolveAutoTypeEnabled(); } case Enable: return true; case Disable: return false; default: Q_ASSERT(false); return false; } }