translation script
This commit is contained in:
parent
e7d366328f
commit
530b46a882
@ -30,7 +30,7 @@
|
|||||||
<url type="donation">https://martinrotter.github.io/donate/</url>
|
<url type="donation">https://martinrotter.github.io/donate/</url>
|
||||||
<content_rating type="oars-1.1" />
|
<content_rating type="oars-1.1" />
|
||||||
<releases>
|
<releases>
|
||||||
<release version="3.9.0" date="2021-03-30"/>
|
<release version="3.9.0" date="2021-03-31"/>
|
||||||
</releases>
|
</releases>
|
||||||
<content_rating type="oars-1.0">
|
<content_rating type="oars-1.0">
|
||||||
<content_attribute id="violence-cartoon">none</content_attribute>
|
<content_attribute id="violence-cartoon">none</content_attribute>
|
||||||
|
39
resources/scripts/scrapers/translate-rss2.py
Executable file
39
resources/scripts/scrapers/translate-rss2.py
Executable file
@ -0,0 +1,39 @@
|
|||||||
|
# Translates entries of RSS 2.0 feed into different locale.
|
||||||
|
#
|
||||||
|
# Make sure to have all dependencies installed:
|
||||||
|
# pip3 install googletrans==4.0.0-rc1
|
||||||
|
#
|
||||||
|
# You must provide raw RSS 2.0 UTF-8 feed XML data as input, for example with curl:
|
||||||
|
# curl 'https://phys.org/rss-feed/' | python ./translate-rss2.py "en" "pt_BR"
|
||||||
|
#
|
||||||
|
# You must provide two additional command line arguments:
|
||||||
|
# translate-rss2.py [FROM-LANGUAGE] [TO-LANGUAGE]
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from googletrans import Translator
|
||||||
|
|
||||||
|
lang_from = sys.argv[1]
|
||||||
|
lang_to = sys.argv[2]
|
||||||
|
sys.stdin.reconfigure(encoding='utf-8')
|
||||||
|
rss_data = sys.stdin.read()
|
||||||
|
rss_document = ET.fromstring(rss_data)
|
||||||
|
translator = Translator()
|
||||||
|
|
||||||
|
def translate_string(to_translate):
|
||||||
|
translated_text = translator.translate(to_translate, src = lang_from, dest = lang_to)
|
||||||
|
time.sleep(0.2)
|
||||||
|
return translated_text.text
|
||||||
|
|
||||||
|
def process_article(article):
|
||||||
|
title = article.find("title")
|
||||||
|
title.text = translate_string(title.text)
|
||||||
|
|
||||||
|
contents = article.find("description")
|
||||||
|
contents.text = translate_string(" ".join(contents.itertext()))
|
||||||
|
|
||||||
|
for article in rss_document.findall(".//item"):
|
||||||
|
process_article(article)
|
||||||
|
|
||||||
|
print(ET.tostring(rss_document, encoding = "unicode"))
|
@ -30,6 +30,7 @@ FeedsProxyModel::FeedsProxyModel(FeedsModel* source_model, QObject* parent)
|
|||||||
RootItem::Kind::Feed,
|
RootItem::Kind::Feed,
|
||||||
RootItem::Kind::Labels,
|
RootItem::Kind::Labels,
|
||||||
RootItem::Kind::Important,
|
RootItem::Kind::Important,
|
||||||
|
RootItem::Kind::Unread,
|
||||||
RootItem::Kind::Bin
|
RootItem::Kind::Bin
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -638,6 +638,31 @@ int DatabaseQueries::getImportantMessageCounts(const QSqlDatabase& db, int accou
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int DatabaseQueries::getUnreadMessageCounts(const QSqlDatabase& db, int account_id, bool* ok) {
|
||||||
|
QSqlQuery q(db);
|
||||||
|
|
||||||
|
q.setForwardOnly(true);
|
||||||
|
q.prepare("SELECT count(*) FROM Messages "
|
||||||
|
"WHERE is_read = 0 AND is_deleted = 0 AND is_pdeleted = 0 AND account_id = :account_id;");
|
||||||
|
|
||||||
|
q.bindValue(QSL(":account_id"), account_id);
|
||||||
|
|
||||||
|
if (q.exec() && q.next()) {
|
||||||
|
if (ok != nullptr) {
|
||||||
|
*ok = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return q.value(0).toInt();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (ok != nullptr) {
|
||||||
|
*ok = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int DatabaseQueries::getMessageCountsForBin(const QSqlDatabase& db, int account_id, bool including_total_counts, bool* ok) {
|
int DatabaseQueries::getMessageCountsForBin(const QSqlDatabase& db, int account_id, bool including_total_counts, bool* ok) {
|
||||||
QSqlQuery q(db);
|
QSqlQuery q(db);
|
||||||
|
|
||||||
@ -781,6 +806,40 @@ QList<Message> DatabaseQueries::getUndeletedImportantMessages(const QSqlDatabase
|
|||||||
return messages;
|
return messages;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QList<Message> DatabaseQueries::getUndeletedUnreadMessages(const QSqlDatabase& db, int account_id, bool* ok) {
|
||||||
|
QList<Message> messages;
|
||||||
|
QSqlQuery q(db);
|
||||||
|
|
||||||
|
q.setForwardOnly(true);
|
||||||
|
q.prepare(QSL("SELECT %1 "
|
||||||
|
"FROM Messages "
|
||||||
|
"WHERE is_read = 0 AND is_deleted = 0 AND "
|
||||||
|
" is_pdeleted = 0 AND account_id = :account_id;").arg(messageTableAttributes(true).values().join(QSL(", "))));
|
||||||
|
q.bindValue(QSL(":account_id"), account_id);
|
||||||
|
|
||||||
|
if (q.exec()) {
|
||||||
|
while (q.next()) {
|
||||||
|
bool decoded;
|
||||||
|
Message message = Message::fromSqlRecord(q.record(), &decoded);
|
||||||
|
|
||||||
|
if (decoded) {
|
||||||
|
messages.append(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ok != nullptr) {
|
||||||
|
*ok = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (ok != nullptr) {
|
||||||
|
*ok = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return messages;
|
||||||
|
}
|
||||||
|
|
||||||
QList<Message> DatabaseQueries::getUndeletedMessagesForFeed(const QSqlDatabase& db, const QString& feed_custom_id,
|
QList<Message> DatabaseQueries::getUndeletedMessagesForFeed(const QSqlDatabase& db, const QString& feed_custom_id,
|
||||||
int account_id, bool* ok) {
|
int account_id, bool* ok) {
|
||||||
QList<Message> messages;
|
QList<Message> messages;
|
||||||
@ -1603,6 +1662,29 @@ QStringList DatabaseQueries::customIdsOfImportantMessages(const QSqlDatabase& db
|
|||||||
return ids;
|
return ids;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QStringList DatabaseQueries::customIdsOfUnreadMessages(const QSqlDatabase& db, int account_id, bool* ok) {
|
||||||
|
QSqlQuery q(db);
|
||||||
|
QStringList ids;
|
||||||
|
|
||||||
|
q.setForwardOnly(true);
|
||||||
|
q.prepare(QSL("SELECT custom_id FROM Messages "
|
||||||
|
"WHERE is_read = 0 AND is_deleted = 0 AND is_pdeleted = 0 AND account_id = :account_id;"));
|
||||||
|
q.bindValue(QSL(":account_id"), account_id);
|
||||||
|
|
||||||
|
if (ok != nullptr) {
|
||||||
|
*ok = q.exec();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
q.exec();
|
||||||
|
}
|
||||||
|
|
||||||
|
while (q.next()) {
|
||||||
|
ids.append(q.value(0).toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
return ids;
|
||||||
|
}
|
||||||
|
|
||||||
QStringList DatabaseQueries::customIdsOfMessagesFromBin(const QSqlDatabase& db, int account_id, bool* ok) {
|
QStringList DatabaseQueries::customIdsOfMessagesFromBin(const QSqlDatabase& db, int account_id, bool* ok) {
|
||||||
QSqlQuery q(db);
|
QSqlQuery q(db);
|
||||||
QStringList ids;
|
QStringList ids;
|
||||||
|
@ -78,12 +78,14 @@ class DatabaseQueries {
|
|||||||
bool only_total_counts, bool* ok = nullptr);
|
bool only_total_counts, bool* ok = nullptr);
|
||||||
static int getImportantMessageCounts(const QSqlDatabase& db, int account_id,
|
static int getImportantMessageCounts(const QSqlDatabase& db, int account_id,
|
||||||
bool only_total_counts, bool* ok = nullptr);
|
bool only_total_counts, bool* ok = nullptr);
|
||||||
|
static int getUnreadMessageCounts(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
static int getMessageCountsForBin(const QSqlDatabase& db, int account_id, bool including_total_counts, bool* ok = nullptr);
|
static int getMessageCountsForBin(const QSqlDatabase& db, int account_id, bool including_total_counts, bool* ok = nullptr);
|
||||||
|
|
||||||
// Get messages (for newspaper view for example).
|
// Get messages (for newspaper view for example).
|
||||||
static QList<Message> getUndeletedMessagesWithLabel(const QSqlDatabase& db, const Label* label, bool* ok = nullptr);
|
static QList<Message> getUndeletedMessagesWithLabel(const QSqlDatabase& db, const Label* label, bool* ok = nullptr);
|
||||||
static QList<Message> getUndeletedLabelledMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
static QList<Message> getUndeletedLabelledMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
static QList<Message> getUndeletedImportantMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
static QList<Message> getUndeletedImportantMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
|
static QList<Message> getUndeletedUnreadMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
static QList<Message> getUndeletedMessagesForFeed(const QSqlDatabase& db, const QString& feed_custom_id,
|
static QList<Message> getUndeletedMessagesForFeed(const QSqlDatabase& db, const QString& feed_custom_id,
|
||||||
int account_id, bool* ok = nullptr);
|
int account_id, bool* ok = nullptr);
|
||||||
static QList<Message> getUndeletedMessagesForBin(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
static QList<Message> getUndeletedMessagesForBin(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
@ -92,6 +94,7 @@ class DatabaseQueries {
|
|||||||
// Custom ID accumulators.
|
// Custom ID accumulators.
|
||||||
static QStringList customIdsOfMessagesFromLabel(const QSqlDatabase& db, Label* label, bool* ok = nullptr);
|
static QStringList customIdsOfMessagesFromLabel(const QSqlDatabase& db, Label* label, bool* ok = nullptr);
|
||||||
static QStringList customIdsOfImportantMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
static QStringList customIdsOfImportantMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
|
static QStringList customIdsOfUnreadMessages(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
static QStringList customIdsOfMessagesFromAccount(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
static QStringList customIdsOfMessagesFromAccount(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
static QStringList customIdsOfMessagesFromBin(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
static QStringList customIdsOfMessagesFromBin(const QSqlDatabase& db, int account_id, bool* ok = nullptr);
|
||||||
static QStringList customIdsOfMessagesFromFeed(const QSqlDatabase& db, const QString& feed_custom_id, int account_id,
|
static QStringList customIdsOfMessagesFromFeed(const QSqlDatabase& db, const QString& feed_custom_id, int account_id,
|
||||||
|
@ -43,6 +43,7 @@
|
|||||||
#define ID_RECYCLE_BIN -2
|
#define ID_RECYCLE_BIN -2
|
||||||
#define ID_IMPORTANT -3
|
#define ID_IMPORTANT -3
|
||||||
#define ID_LABELS -4
|
#define ID_LABELS -4
|
||||||
|
#define ID_UNREAD -5
|
||||||
|
|
||||||
#define MSG_SCORE_MAX 100.0
|
#define MSG_SCORE_MAX 100.0
|
||||||
#define MSG_SCORE_MIN 0.0
|
#define MSG_SCORE_MIN 0.0
|
||||||
|
@ -747,7 +747,8 @@ void FeedsView::contextMenuEvent(QContextMenuEvent* event) {
|
|||||||
// Display context menu for feeds.
|
// Display context menu for feeds.
|
||||||
initializeContextMenuFeeds(clicked_item)->exec(event->globalPos());
|
initializeContextMenuFeeds(clicked_item)->exec(event->globalPos());
|
||||||
}
|
}
|
||||||
else if (clicked_item->kind() == RootItem::Kind::Important) {
|
else if (clicked_item->kind() == RootItem::Kind::Important ||
|
||||||
|
clicked_item->kind() == RootItem::Kind::Unread) {
|
||||||
initializeContextMenuImportant(clicked_item)->exec(event->globalPos());
|
initializeContextMenuImportant(clicked_item)->exec(event->globalPos());
|
||||||
}
|
}
|
||||||
else if (clicked_item->kind() == RootItem::Kind::Bin) {
|
else if (clicked_item->kind() == RootItem::Kind::Bin) {
|
||||||
|
@ -152,6 +152,7 @@ HEADERS += core/feeddownloader.h \
|
|||||||
services/abstract/rootitem.h \
|
services/abstract/rootitem.h \
|
||||||
services/abstract/serviceentrypoint.h \
|
services/abstract/serviceentrypoint.h \
|
||||||
services/abstract/serviceroot.h \
|
services/abstract/serviceroot.h \
|
||||||
|
services/abstract/unreadnode.h \
|
||||||
services/feedly/definitions.h \
|
services/feedly/definitions.h \
|
||||||
services/feedly/feedlyentrypoint.h \
|
services/feedly/feedlyentrypoint.h \
|
||||||
services/feedly/feedlynetwork.h \
|
services/feedly/feedlynetwork.h \
|
||||||
@ -326,6 +327,7 @@ SOURCES += core/feeddownloader.cpp \
|
|||||||
services/abstract/recyclebin.cpp \
|
services/abstract/recyclebin.cpp \
|
||||||
services/abstract/rootitem.cpp \
|
services/abstract/rootitem.cpp \
|
||||||
services/abstract/serviceroot.cpp \
|
services/abstract/serviceroot.cpp \
|
||||||
|
services/abstract/unreadnode.cpp \
|
||||||
services/feedly/feedlyentrypoint.cpp \
|
services/feedly/feedlyentrypoint.cpp \
|
||||||
services/feedly/feedlynetwork.cpp \
|
services/feedly/feedlynetwork.cpp \
|
||||||
services/feedly/feedlyserviceroot.cpp \
|
services/feedly/feedlyserviceroot.cpp \
|
||||||
|
@ -313,6 +313,7 @@ bool AccountCheckSortedModel::lessThan(const QModelIndex& source_left, const QMo
|
|||||||
RootItem::Kind::Feed,
|
RootItem::Kind::Feed,
|
||||||
RootItem::Kind::Labels,
|
RootItem::Kind::Labels,
|
||||||
RootItem::Kind::Important,
|
RootItem::Kind::Important,
|
||||||
|
RootItem::Kind::Unread,
|
||||||
RootItem::Kind::Bin
|
RootItem::Kind::Bin
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
#include "services/abstract/labelsnode.h"
|
#include "services/abstract/labelsnode.h"
|
||||||
#include "services/abstract/recyclebin.h"
|
#include "services/abstract/recyclebin.h"
|
||||||
#include "services/abstract/serviceroot.h"
|
#include "services/abstract/serviceroot.h"
|
||||||
|
#include "services/abstract/unreadnode.h"
|
||||||
|
|
||||||
#include <QThread>
|
#include <QThread>
|
||||||
|
|
||||||
@ -233,6 +234,11 @@ int Feed::updateMessages(const QList<Message>& messages, bool error_during_obtai
|
|||||||
items_to_update.append(getParentServiceRoot()->importantNode());
|
items_to_update.append(getParentServiceRoot()->importantNode());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (getParentServiceRoot()->unreadNode() != nullptr && anything_updated) {
|
||||||
|
getParentServiceRoot()->unreadNode()->updateCounts(true);
|
||||||
|
items_to_update.append(getParentServiceRoot()->unreadNode());
|
||||||
|
}
|
||||||
|
|
||||||
if (getParentServiceRoot()->labelsNode() != nullptr) {
|
if (getParentServiceRoot()->labelsNode() != nullptr) {
|
||||||
getParentServiceRoot()->labelsNode()->updateCounts(true);
|
getParentServiceRoot()->labelsNode()->updateCounts(true);
|
||||||
items_to_update.append(getParentServiceRoot()->labelsNode());
|
items_to_update.append(getParentServiceRoot()->labelsNode());
|
||||||
|
@ -10,7 +10,6 @@ class ImportantNode : public RootItem {
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
explicit ImportantNode(RootItem* parent_item = nullptr);
|
explicit ImportantNode(RootItem* parent_item = nullptr);
|
||||||
virtual ~ImportantNode() = default;
|
|
||||||
|
|
||||||
virtual QList<Message> undeletedMessages() const;
|
virtual QList<Message> undeletedMessages() const;
|
||||||
virtual bool cleanMessages(bool clean_read_only);
|
virtual bool cleanMessages(bool clean_read_only);
|
||||||
|
@ -207,13 +207,21 @@ bool RootItem::performDragDropChange(RootItem* target_item) {
|
|||||||
|
|
||||||
int RootItem::countOfUnreadMessages() const {
|
int RootItem::countOfUnreadMessages() const {
|
||||||
return boolinq::from(m_childItems).sum([](RootItem* it) {
|
return boolinq::from(m_childItems).sum([](RootItem* it) {
|
||||||
return (it->kind() == RootItem::Kind::Important || it->kind() == RootItem::Kind::Labels) ? 0 : it->countOfUnreadMessages();
|
return (it->kind() == RootItem::Kind::Important ||
|
||||||
|
it->kind() == RootItem::Kind::Unread ||
|
||||||
|
it->kind() == RootItem::Kind::Labels)
|
||||||
|
? 0
|
||||||
|
: it->countOfUnreadMessages();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
int RootItem::countOfAllMessages() const {
|
int RootItem::countOfAllMessages() const {
|
||||||
return boolinq::from(m_childItems).sum([](RootItem* it) {
|
return boolinq::from(m_childItems).sum([](RootItem* it) {
|
||||||
return (it->kind() == RootItem::Kind::Important || it->kind() == RootItem::Kind::Labels) ? 0 : it->countOfAllMessages();
|
return (it->kind() == RootItem::Kind::Important ||
|
||||||
|
it->kind() == RootItem::Kind::Unread ||
|
||||||
|
it->kind() == RootItem::Kind::Labels)
|
||||||
|
? 0
|
||||||
|
: it->countOfAllMessages();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -47,7 +47,8 @@ class RSSGUARD_DLLSPEC RootItem : public QObject {
|
|||||||
ServiceRoot = 16,
|
ServiceRoot = 16,
|
||||||
Labels = 32,
|
Labels = 32,
|
||||||
Important = 64,
|
Important = 64,
|
||||||
Label = 128
|
Label = 128,
|
||||||
|
Unread = 256
|
||||||
};
|
};
|
||||||
|
|
||||||
// Constructors and destructors.
|
// Constructors and destructors.
|
||||||
|
@ -16,10 +16,12 @@
|
|||||||
#include "services/abstract/importantnode.h"
|
#include "services/abstract/importantnode.h"
|
||||||
#include "services/abstract/labelsnode.h"
|
#include "services/abstract/labelsnode.h"
|
||||||
#include "services/abstract/recyclebin.h"
|
#include "services/abstract/recyclebin.h"
|
||||||
|
#include "services/abstract/unreadnode.h"
|
||||||
|
|
||||||
ServiceRoot::ServiceRoot(RootItem* parent)
|
ServiceRoot::ServiceRoot(RootItem* parent)
|
||||||
: RootItem(parent), m_recycleBin(new RecycleBin(this)), m_importantNode(new ImportantNode(this)),
|
: RootItem(parent), m_recycleBin(new RecycleBin(this)), m_importantNode(new ImportantNode(this)),
|
||||||
m_labelsNode(new LabelsNode(this)), m_accountId(NO_PARENT_CATEGORY), m_networkProxy(QNetworkProxy()) {
|
m_labelsNode(new LabelsNode(this)), m_unreadNode(new UnreadNode(this)),
|
||||||
|
m_accountId(NO_PARENT_CATEGORY), m_networkProxy(QNetworkProxy()) {
|
||||||
setKind(RootItem::Kind::ServiceRoot);
|
setKind(RootItem::Kind::ServiceRoot);
|
||||||
appendCommonNodes();
|
appendCommonNodes();
|
||||||
}
|
}
|
||||||
@ -198,6 +200,7 @@ void ServiceRoot::cleanAllItemsFromModel() {
|
|||||||
for (RootItem* top_level_item : qAsConst(chi)) {
|
for (RootItem* top_level_item : qAsConst(chi)) {
|
||||||
if (top_level_item->kind() != RootItem::Kind::Bin &&
|
if (top_level_item->kind() != RootItem::Kind::Bin &&
|
||||||
top_level_item->kind() != RootItem::Kind::Important &&
|
top_level_item->kind() != RootItem::Kind::Important &&
|
||||||
|
top_level_item->kind() != RootItem::Kind::Unread &&
|
||||||
top_level_item->kind() != RootItem::Kind::Labels) {
|
top_level_item->kind() != RootItem::Kind::Labels) {
|
||||||
requestItemRemoval(top_level_item);
|
requestItemRemoval(top_level_item);
|
||||||
}
|
}
|
||||||
@ -221,6 +224,10 @@ void ServiceRoot::appendCommonNodes() {
|
|||||||
appendChild(importantNode());
|
appendChild(importantNode());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (unreadNode() != nullptr && !childItems().contains(unreadNode())) {
|
||||||
|
appendChild(unreadNode());
|
||||||
|
}
|
||||||
|
|
||||||
if (labelsNode() != nullptr && !childItems().contains(labelsNode())) {
|
if (labelsNode() != nullptr && !childItems().contains(labelsNode())) {
|
||||||
appendChild(labelsNode());
|
appendChild(labelsNode());
|
||||||
}
|
}
|
||||||
@ -391,6 +398,10 @@ LabelsNode* ServiceRoot::labelsNode() const {
|
|||||||
return m_labelsNode;
|
return m_labelsNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
UnreadNode* ServiceRoot::unreadNode() const {
|
||||||
|
return m_unreadNode;
|
||||||
|
}
|
||||||
|
|
||||||
void ServiceRoot::syncIn() {
|
void ServiceRoot::syncIn() {
|
||||||
QIcon original_icon = icon();
|
QIcon original_icon = icon();
|
||||||
|
|
||||||
@ -516,6 +527,13 @@ QStringList ServiceRoot::customIDSOfMessagesForItem(RootItem* item) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case RootItem::Kind::Unread: {
|
||||||
|
QSqlDatabase database = qApp->database()->driver()->connection(metaObject()->className());
|
||||||
|
|
||||||
|
list = DatabaseQueries::customIdsOfUnreadMessages(database, accountId());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -606,6 +624,10 @@ bool ServiceRoot::loadMessagesForItem(RootItem* item, MessagesModel* model) {
|
|||||||
model->setFilter(QString("Messages.is_important = 1 AND Messages.is_deleted = 0 AND Messages.is_pdeleted = 0 AND Messages.account_id = %1")
|
model->setFilter(QString("Messages.is_important = 1 AND Messages.is_deleted = 0 AND Messages.is_pdeleted = 0 AND Messages.account_id = %1")
|
||||||
.arg(QString::number(accountId())));
|
.arg(QString::number(accountId())));
|
||||||
}
|
}
|
||||||
|
else if (item->kind() == RootItem::Kind::Unread) {
|
||||||
|
model->setFilter(QString("Messages.is_read = 0 AND Messages.is_deleted = 0 AND Messages.is_pdeleted = 0 AND Messages.account_id = %1")
|
||||||
|
.arg(QString::number(accountId())));
|
||||||
|
}
|
||||||
else if (item->kind() == RootItem::Kind::Label) {
|
else if (item->kind() == RootItem::Kind::Label) {
|
||||||
// Show messages with particular label.
|
// Show messages with particular label.
|
||||||
model->setFilter(QString("Messages.is_deleted = 0 AND Messages.is_pdeleted = 0 AND Messages.account_id = %1 AND "
|
model->setFilter(QString("Messages.is_deleted = 0 AND Messages.is_pdeleted = 0 AND Messages.account_id = %1 AND "
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
class FeedsModel;
|
class FeedsModel;
|
||||||
class RecycleBin;
|
class RecycleBin;
|
||||||
class ImportantNode;
|
class ImportantNode;
|
||||||
|
class UnreadNode;
|
||||||
class LabelsNode;
|
class LabelsNode;
|
||||||
class Label;
|
class Label;
|
||||||
class QAction;
|
class QAction;
|
||||||
@ -43,6 +44,7 @@ class ServiceRoot : public RootItem {
|
|||||||
RecycleBin* recycleBin() const;
|
RecycleBin* recycleBin() const;
|
||||||
ImportantNode* importantNode() const;
|
ImportantNode* importantNode() const;
|
||||||
LabelsNode* labelsNode() const;
|
LabelsNode* labelsNode() const;
|
||||||
|
UnreadNode* unreadNode() const;
|
||||||
|
|
||||||
virtual void updateCounts(bool including_total_count);
|
virtual void updateCounts(bool including_total_count);
|
||||||
virtual bool canBeDeleted() const;
|
virtual bool canBeDeleted() const;
|
||||||
@ -256,6 +258,7 @@ class ServiceRoot : public RootItem {
|
|||||||
RecycleBin* m_recycleBin;
|
RecycleBin* m_recycleBin;
|
||||||
ImportantNode* m_importantNode;
|
ImportantNode* m_importantNode;
|
||||||
LabelsNode* m_labelsNode;
|
LabelsNode* m_labelsNode;
|
||||||
|
UnreadNode* m_unreadNode;
|
||||||
int m_accountId;
|
int m_accountId;
|
||||||
QList<QAction*> m_serviceMenu;
|
QList<QAction*> m_serviceMenu;
|
||||||
QNetworkProxy m_networkProxy;
|
QNetworkProxy m_networkProxy;
|
||||||
|
79
src/librssguard/services/abstract/unreadnode.cpp
Executable file
79
src/librssguard/services/abstract/unreadnode.cpp
Executable file
@ -0,0 +1,79 @@
|
|||||||
|
// For license of this file, see <project-root-folder>/LICENSE.md.
|
||||||
|
|
||||||
|
#include "services/abstract/unreadnode.h"
|
||||||
|
|
||||||
|
#include "database/databasequeries.h"
|
||||||
|
#include "miscellaneous/application.h"
|
||||||
|
#include "miscellaneous/iconfactory.h"
|
||||||
|
|
||||||
|
#include <QThread>
|
||||||
|
|
||||||
|
UnreadNode::UnreadNode(RootItem* parent_item) : RootItem(parent_item) {
|
||||||
|
setKind(RootItem::Kind::Unread);
|
||||||
|
setId(ID_UNREAD);
|
||||||
|
setIcon(qApp->icons()->fromTheme(QSL("mail-mark-unread")));
|
||||||
|
setTitle(tr("Unread messages"));
|
||||||
|
setDescription(tr("You can find all unread messages here."));
|
||||||
|
}
|
||||||
|
|
||||||
|
QList<Message> UnreadNode::undeletedMessages() const {
|
||||||
|
QSqlDatabase database = qApp->database()->driver()->connection(metaObject()->className());
|
||||||
|
|
||||||
|
return DatabaseQueries::getUndeletedUnreadMessages(database, getParentServiceRoot()->accountId());
|
||||||
|
}
|
||||||
|
|
||||||
|
void UnreadNode::updateCounts(bool including_total_count) {
|
||||||
|
Q_UNUSED(including_total_count)
|
||||||
|
|
||||||
|
bool is_main_thread = QThread::currentThread() == qApp->thread();
|
||||||
|
QSqlDatabase database = is_main_thread ?
|
||||||
|
qApp->database()->driver()->connection(metaObject()->className()) :
|
||||||
|
qApp->database()->driver()->connection(QSL("feed_upd"));
|
||||||
|
int account_id = getParentServiceRoot()->accountId();
|
||||||
|
|
||||||
|
m_totalCount = m_unreadCount = DatabaseQueries::getUnreadMessageCounts(database, account_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool UnreadNode::cleanMessages(bool clean_read_only) {
|
||||||
|
ServiceRoot* service = getParentServiceRoot();
|
||||||
|
QSqlDatabase database = qApp->database()->driver()->connection(metaObject()->className());
|
||||||
|
|
||||||
|
if (DatabaseQueries::cleanImportantMessages(database, clean_read_only, service->accountId())) {
|
||||||
|
service->updateCounts(true);
|
||||||
|
service->itemChanged(service->getSubTree());
|
||||||
|
service->requestReloadMessageList(true);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool UnreadNode::markAsReadUnread(RootItem::ReadStatus status) {
|
||||||
|
ServiceRoot* service = getParentServiceRoot();
|
||||||
|
auto* cache = dynamic_cast<CacheForServiceRoot*>(service);
|
||||||
|
|
||||||
|
if (cache != nullptr) {
|
||||||
|
cache->addMessageStatesToCache(service->customIDSOfMessagesForItem(this), status);
|
||||||
|
}
|
||||||
|
|
||||||
|
QSqlDatabase database = qApp->database()->driver()->connection(metaObject()->className());
|
||||||
|
|
||||||
|
if (DatabaseQueries::markImportantMessagesReadUnread(database, service->accountId(), status)) {
|
||||||
|
service->updateCounts(false);
|
||||||
|
service->itemChanged(service->getSubTree());
|
||||||
|
service->requestReloadMessageList(status == RootItem::ReadStatus::Read);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int UnreadNode::countOfUnreadMessages() const {
|
||||||
|
return m_unreadCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
int UnreadNode::countOfAllMessages() const {
|
||||||
|
return m_totalCount;
|
||||||
|
}
|
24
src/librssguard/services/abstract/unreadnode.h
Executable file
24
src/librssguard/services/abstract/unreadnode.h
Executable file
@ -0,0 +1,24 @@
|
|||||||
|
// For license of this file, see <project-root-folder>/LICENSE.md.
|
||||||
|
|
||||||
|
#ifndef UNREADNODE_H
|
||||||
|
#define UNREADNODE_H
|
||||||
|
|
||||||
|
#include "services/abstract/rootitem.h"
|
||||||
|
|
||||||
|
class UnreadNode : public RootItem {
|
||||||
|
public:
|
||||||
|
explicit UnreadNode(RootItem* parent_item = nullptr);
|
||||||
|
|
||||||
|
virtual QList<Message> undeletedMessages() const;
|
||||||
|
virtual bool cleanMessages(bool clean_read_only);
|
||||||
|
virtual void updateCounts(bool including_total_count);
|
||||||
|
virtual bool markAsReadUnread(ReadStatus status);
|
||||||
|
virtual int countOfUnreadMessages() const;
|
||||||
|
virtual int countOfAllMessages() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_totalCount{};
|
||||||
|
int m_unreadCount{};
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // UNREADNODE_H
|
Loading…
x
Reference in New Issue
Block a user