#include "gui/feedsview.h" #include "core/defs.h" #include "core/feedsmodelfeed.h" #include "core/feedsmodel.h" #include "core/feedsproxymodel.h" #include "core/feedsmodelrootitem.h" #include "core/feedsmodelcategory.h" #include "core/feedsmodelstandardcategory.h" #include "gui/formmain.h" #include "gui/formstandardcategorydetails.h" #include #include #include #include #include FeedsView::FeedsView(QWidget *parent) : QTreeView(parent), m_contextMenuCategoriesFeeds(NULL), m_contextMenuEmptySpace(NULL) { m_proxyModel = new FeedsProxyModel(this); m_sourceModel = m_proxyModel->sourceModel(); setModel(m_proxyModel); setupAppearance(); } FeedsView::~FeedsView() { qDebug("Destroying FeedsView instance."); } void FeedsView::setSortingEnabled(bool enable) { QTreeView::setSortingEnabled(enable); header()->setSortIndicatorShown(false); } QList FeedsView::selectedFeeds() const { QModelIndexList selection = selectionModel()->selectedRows(); QModelIndexList mapped_selection = m_proxyModel->mapListToSource(selection); return m_sourceModel->feedsForIndexes(mapped_selection); } QList FeedsView::allFeeds() const { return m_sourceModel->allFeeds(); } FeedsModelCategory *FeedsView::isCurrentIndexCategory() const { QModelIndex current_mapped = m_proxyModel->mapToSource(currentIndex()); return m_sourceModel->categoryForIndex(current_mapped); } FeedsModelFeed *FeedsView::isCurrentIndexFeed() const { QModelIndex current_mapped = m_proxyModel->mapToSource(currentIndex()); return m_sourceModel->feedForIndex(current_mapped); } void FeedsView::setSelectedFeedsClearStatus(int clear) { m_sourceModel->markFeedsDeleted(selectedFeeds(), clear); updateCountsOfSelectedFeeds(); emit feedsNeedToBeReloaded(1); } void FeedsView::clearSelectedFeeds() { setSelectedFeedsClearStatus(1); } void FeedsView::addNewStandardCategory() { QPointer form_pointer = new FormStandardCategoryDetails(m_sourceModel, this); if (form_pointer.data()->exec(NULL) == QDialog::Accepted) { // TODO: nova kategorie pridana } else { // TODO: nova kategorie nepridana } delete form_pointer.data(); } void FeedsView::editStandardCategory(FeedsModelStandardCategory *category) { FeedsModelStandardCategory *std_category = static_cast(category); QPointer form_pointer = new FormStandardCategoryDetails(m_sourceModel, this); if (form_pointer.data()->exec(std_category) == QDialog::Accepted) { // TODO: kategorie upravena } else { // TODO: kategorie neupravena (uživatel zrušil dialog) } delete form_pointer.data(); } void FeedsView::editSelectedItem() { // TODO: preda pridavanim/upravou/mazanim kanalu/kategorii // ziskat ZAPISOVACI zamek pres systemfactory::applicationCloseLock // a po dokonceni cinnosti jej odevzdavat FeedsModelCategory *category; FeedsModelFeed *feed; if ((category = isCurrentIndexCategory()) != NULL) { // Category is selected. switch (category->type()) { case FeedsModelCategory::Standard: { // User wants to edit standard category. editStandardCategory(static_cast(category)); break; } default: break; } } else if ((feed = isCurrentIndexFeed()) != NULL) { // Feed is selected. } } void FeedsView::deleteSelectedItem() { QModelIndex current_index = currentIndex(); QItemSelectionModel *selection_model = selectionModel(); if (!current_index.isValid()) { return; } if (selection_model->selectedRows().size() > 1) { // pridat BalloonTip z qonverteru a tady // ho odpalit pokud todle nastane // s hlaskou "More than one item selected, removing // only current one." selection_model->clearSelection(); selection_model->select(current_index, QItemSelectionModel::Rows | QItemSelectionModel::SelectCurrent); } m_sourceModel->removeItem(m_proxyModel->mapToSource(current_index)); } void FeedsView::markSelectedFeedsReadStatus(int read) { m_sourceModel->markFeedsRead(selectedFeeds(), read); updateCountsOfSelectedFeeds(false); emit feedsNeedToBeReloaded(read); } void FeedsView::markSelectedFeedsRead() { markSelectedFeedsReadStatus(1); } void FeedsView::markSelectedFeedsUnread() { markSelectedFeedsReadStatus(0); } void FeedsView::markAllFeedsReadStatus(int read) { m_sourceModel->markFeedsRead(allFeeds(), read); updateCountsOfAllFeeds(false); emit feedsNeedToBeReloaded(read); } void FeedsView::markAllFeedsRead() { markAllFeedsReadStatus(1); } void FeedsView::openSelectedFeedsInNewspaperMode() { QList messages = m_sourceModel->messagesForFeeds(selectedFeeds()); if (!messages.isEmpty()) { emit openMessagesInNewspaperView(messages); } } void FeedsView::updateCountsOfSelectedFeeds(bool update_total_too) { QList selected_feeds = selectedFeeds(); if (!selected_feeds.isEmpty()) { foreach (FeedsModelFeed *feed, selected_feeds) { feed->updateCounts(update_total_too); } // Make sure that selected view reloads changed indexes. m_sourceModel->reloadChangedLayout(m_proxyModel->mapListToSource(selectionModel()->selectedRows())); notifyWithCounts(); } } void FeedsView::updateCountsOfAllFeeds(bool update_total_too) { foreach (FeedsModelFeed *feed, allFeeds()) { feed->updateCounts(update_total_too); } // Make sure that all views reloads its data. m_sourceModel->reloadWholeLayout(); notifyWithCounts(); } void FeedsView::updateCountsOfParticularFeed(FeedsModelFeed *feed, bool update_total_too) { QModelIndex index = m_sourceModel->indexForItem(feed); if (index.isValid()) { feed->updateCounts(update_total_too); m_sourceModel->reloadChangedLayout(QModelIndexList() << index); } notifyWithCounts(); } void FeedsView::initializeContextMenuCategoriesFeeds() { m_contextMenuCategoriesFeeds = new QMenu(tr("Context menu for feeds"), this); m_contextMenuCategoriesFeeds->addActions(QList() << FormMain::instance()->m_ui->m_actionUpdateSelectedFeedsCategories << FormMain::instance()->m_ui->m_actionViewSelectedItemsNewspaperMode << FormMain::instance()->m_ui->m_actionMarkFeedsAsRead << FormMain::instance()->m_ui->m_actionMarkFeedsAsUnread); } void FeedsView::initializeContextMenuEmptySpace() { m_contextMenuEmptySpace = new QMenu(tr("Context menu for feeds"), this); m_contextMenuEmptySpace->addActions(QList() << FormMain::instance()->m_ui->m_actionUpdateAllFeeds); } void FeedsView::setupAppearance() { #if QT_VERSION >= 0x050000 // Setup column resize strategies. header()->setSectionResizeMode(FDS_MODEL_TITLE_INDEX, QHeaderView::Stretch); header()->setSectionResizeMode(FDS_MODEL_COUNTS_INDEX, QHeaderView::ResizeToContents); #else // Setup column resize strategies. header()->setResizeMode(FDS_MODEL_TITLE_INDEX, QHeaderView::Stretch); header()->setResizeMode(FDS_MODEL_COUNTS_INDEX, QHeaderView::ResizeToContents); #endif header()->setStretchLastSection(false); setUniformRowHeights(true); setAcceptDrops(false); setDragEnabled(false); setAnimated(true); setSortingEnabled(true); setItemsExpandable(true); setExpandsOnDoubleClick(true); setEditTriggers(QAbstractItemView::NoEditTriggers); setIndentation(10); setDragDropMode(QAbstractItemView::NoDragDrop); setAllColumnsShowFocus(true); setSelectionMode(QAbstractItemView::ExtendedSelection); setRootIsDecorated(false); // Sort in ascending order, that is categories are // "bigger" than feeds. sortByColumn(0, Qt::AscendingOrder); } void FeedsView::selectionChanged(const QItemSelection &selected, const QItemSelection &deselected) { QTreeView::selectionChanged(selected, deselected); m_selectedFeeds.clear(); foreach (FeedsModelFeed *feed, selectedFeeds()) { #if defined(DEBUG) QModelIndex index_for_feed = m_sourceModel->indexForItem(feed); qDebug("Selecting feed '%s' (source index [%d, %d]).", qPrintable(feed->title()), index_for_feed.row(), index_for_feed.column()); #endif m_selectedFeeds << feed->id(); } emit feedsSelected(m_selectedFeeds); } void FeedsView::keyPressEvent(QKeyEvent *event) { QTreeView::keyPressEvent(event); if (event->key() == Qt::Key_Delete) { deleteSelectedItem(); } } void FeedsView::contextMenuEvent(QContextMenuEvent *event) { if (indexAt(event->pos()).isValid()) { // Display context menu for categories. if (m_contextMenuCategoriesFeeds == NULL) { // Context menu is not initialized, initialize. initializeContextMenuCategoriesFeeds(); } m_contextMenuCategoriesFeeds->exec(event->globalPos()); } else { // Display menu for empty space. if (m_contextMenuEmptySpace == NULL) { // Context menu is not initialized, initialize. initializeContextMenuEmptySpace(); } m_contextMenuEmptySpace->exec(event->globalPos()); } }