#include "vnotebookselector.h" #include #include #include #include #include #include #include #include #include #include #include "vnotebook.h" #include "vconfigmanager.h" #include "dialog/vnewnotebookdialog.h" #include "dialog/vnotebookinfodialog.h" #include "dialog/vdeletenotebookdialog.h" #include "vnotebook.h" #include "vdirectory.h" #include "utils/vutils.h" #include "vnote.h" #include "veditarea.h" #include "vnofocusitemdelegate.h" extern VConfigManager vconfig; extern VNote *g_vnote; const int VNotebookSelector::c_notebookStartIdx = 1; VNotebookSelector::VNotebookSelector(VNote *vnote, QWidget *p_parent) : QComboBox(p_parent), VNavigationMode(), m_vnote(vnote), m_notebooks(m_vnote->getNotebooks()), m_editArea(NULL), m_lastValidIndex(-1), m_naviLabel(NULL) { m_listWidget = new QListWidget(this); m_listWidget->setItemDelegate(new VNoFocusItemDelegate(this)); m_listWidget->setContextMenuPolicy(Qt::CustomContextMenu); connect(m_listWidget, &QListWidget::customContextMenuRequested, this, &VNotebookSelector::requestPopupListContextMenu); setModel(m_listWidget->model()); setView(m_listWidget); m_listWidget->viewport()->installEventFilter(this); m_listWidget->installEventFilter(this); initActions(); connect(this, SIGNAL(currentIndexChanged(int)), this, SLOT(handleCurIndexChanged(int))); connect(this, SIGNAL(activated(int)), this, SLOT(handleItemActivated(int))); } void VNotebookSelector::initActions() { m_deleteNotebookAct = new QAction(QIcon(":/resources/icons/delete_notebook.svg"), tr("&Delete"), this); m_deleteNotebookAct->setToolTip(tr("Delete current notebook")); connect(m_deleteNotebookAct, SIGNAL(triggered(bool)), this, SLOT(deleteNotebook())); m_notebookInfoAct = new QAction(QIcon(":/resources/icons/notebook_info.svg"), tr("&Info"), this); m_notebookInfoAct->setToolTip(tr("View and edit current notebook's information")); connect(m_notebookInfoAct, SIGNAL(triggered(bool)), this, SLOT(editNotebookInfo())); } void VNotebookSelector::updateComboBox() { int index = vconfig.getCurNotebookIndex(); disconnect(this, SIGNAL(currentIndexChanged(int)), this, SLOT(handleCurIndexChanged(int))); clear(); m_listWidget->clear(); insertAddNotebookItem(); for (int i = 0; i < m_notebooks.size(); ++i) { addNotebookItem(m_notebooks[i]->getName()); } setCurrentIndex(-1); connect(this, SIGNAL(currentIndexChanged(int)), this, SLOT(handleCurIndexChanged(int))); if (m_notebooks.isEmpty()) { vconfig.setCurNotebookIndex(-1); setCurrentIndex(0); } else { setCurrentIndexNotebook(index); } qDebug() << "notebooks" << m_notebooks.size() << "current index" << index; } void VNotebookSelector::setCurrentIndexNotebook(int p_index) { if (p_index > -1) { p_index += c_notebookStartIdx; } setCurrentIndex(p_index); } void VNotebookSelector::insertAddNotebookItem() { QListWidgetItem *item = new QListWidgetItem(); item->setIcon(QIcon(":/resources/icons/create_notebook.svg")); item->setText(tr("Add Notebook")); QFont font; font.setItalic(true); item->setData(Qt::FontRole, font); item->setToolTip(tr("Create or import a notebook")); m_listWidget->insertItem(0, item); } void VNotebookSelector::handleCurIndexChanged(int p_index) { qDebug() << "current index changed" << p_index << "startIdx" << c_notebookStartIdx; VNotebook *nb = NULL; if (p_index > -1) { if (p_index < c_notebookStartIdx) { // Click a special action item. if (m_listWidget->count() == c_notebookStartIdx) { // There is no regular notebook item. Just let it be selected. p_index = -1; } else { // handleItemActivated() will handle the logics. return; } } else { int nbIdx = p_index - c_notebookStartIdx; Q_ASSERT(nbIdx >= 0); nb = m_notebooks[nbIdx]; } } m_lastValidIndex = p_index; QString tooltip; if (p_index > -1) { p_index -= c_notebookStartIdx; tooltip = nb->getName(); } setToolTip(tooltip); vconfig.setCurNotebookIndex(p_index); emit curNotebookChanged(nb); } void VNotebookSelector::handleItemActivated(int p_index) { if (p_index > -1 && p_index < c_notebookStartIdx) { // Click a special action item if (m_lastValidIndex > -1) { setCurrentIndex(m_lastValidIndex); } newNotebook(); } } void VNotebookSelector::update() { updateComboBox(); } bool VNotebookSelector::newNotebook() { QString info(tr("Please type the name of the notebook and " "choose a folder as the Root Folder of the notebook.")); info += "\n"; info += tr("* The root folder should be used EXCLUSIVELY by VNote and " "it is recommended to be EMPTY."); info += "\n"; info += tr("* A previously created notebook could be imported into VNote " "by choosing its root folder."); QString defaultName; QString defaultPath; // Use empty default name and path to let the dialog to auto generate a name // under the default VNote notebook folder. VNewNotebookDialog dialog(tr("Add Notebook"), info, defaultName, defaultPath, m_notebooks, this); if (dialog.exec() == QDialog::Accepted) { createNotebook(dialog.getNameInput(), dialog.getPathInput(), dialog.isImportExistingNotebook(), dialog.getImageFolder()); return true; } return false; } void VNotebookSelector::createNotebook(const QString &p_name, const QString &p_path, bool p_import, const QString &p_imageFolder) { VNotebook *nb = VNotebook::createNotebook(p_name, p_path, p_import, p_imageFolder, m_vnote); if (!nb) { VUtils::showMessage(QMessageBox::Warning, tr("Warning"), tr("Fail to create notebook " "%2 in %3.") .arg(vconfig.c_dataTextStyle).arg(p_name).arg(p_path), "", QMessageBox::Ok, QMessageBox::Ok, this); return; } m_notebooks.append(nb); vconfig.setNotebooks(m_notebooks); addNotebookItem(nb->getName()); setCurrentIndexNotebook(m_notebooks.size() - 1); } void VNotebookSelector::deleteNotebook() { QList items = m_listWidget->selectedItems(); if (items.isEmpty()) { return; } Q_ASSERT(items.size() == 1); QListWidgetItem *item = items[0]; int index = indexOfListItem(item); VNotebook *notebook = getNotebookFromComboIndex(index); Q_ASSERT(notebook); VDeleteNotebookDialog dialog(tr("Delete Notebook"), notebook->getName(), notebook->getPath(), this); if (dialog.exec() == QDialog::Accepted) { bool deleteFiles = dialog.getDeleteFiles(); m_editArea->closeFile(notebook, true); deleteNotebook(notebook, deleteFiles); } } void VNotebookSelector::deleteNotebook(VNotebook *p_notebook, bool p_deleteFiles) { V_ASSERT(p_notebook); int idx = indexOfNotebook(p_notebook); m_notebooks.remove(idx); vconfig.setNotebooks(m_notebooks); removeNotebookItem(idx); QString name(p_notebook->getName()); QString path(p_notebook->getPath()); bool ret = VNotebook::deleteNotebook(p_notebook, p_deleteFiles); if (!ret) { // Notebook could not be deleted completely. int cho = VUtils::showMessage(QMessageBox::Information, tr("Delete Notebook Folder From Disk"), tr("Fail to delete the root folder of notebook " "%2 from disk. You may open " "the folder and check it manually.") .arg(vconfig.c_dataTextStyle).arg(name), "", QMessageBox::Open | QMessageBox::Ok, QMessageBox::Ok, this); if (cho == QMessageBox::Open) { // Open the notebook location. QUrl url = QUrl::fromLocalFile(path); QDesktopServices::openUrl(url); } } } int VNotebookSelector::indexOfNotebook(const VNotebook *p_notebook) { for (int i = 0; i < m_notebooks.size(); ++i) { if (m_notebooks[i] == p_notebook) { return i; } } return -1; } void VNotebookSelector::editNotebookInfo() { QList items = m_listWidget->selectedItems(); if (items.isEmpty()) { return; } Q_ASSERT(items.size() == 1); QListWidgetItem *item = items[0]; int index = indexOfListItem(item); VNotebook *notebook = getNotebookFromComboIndex(index); QString curName = notebook->getName(); VNotebookInfoDialog dialog(tr("Notebook Information"), "", notebook, m_notebooks, this); if (dialog.exec() == QDialog::Accepted) { bool updated = false; QString name = dialog.getName(); if (name != curName) { updated = true; notebook->rename(name); updateComboBoxItem(index, name); vconfig.setNotebooks(m_notebooks); } QString imageFolder = dialog.getImageFolder(); if (imageFolder != notebook->getImageFolderConfig()) { updated = true; notebook->setImageFolder(imageFolder); notebook->writeConfig(); } if (updated) { emit notebookUpdated(notebook); } } } void VNotebookSelector::addNotebookItem(const QString &p_name) { QListWidgetItem *item = new QListWidgetItem(m_listWidget); item->setText(p_name); item->setToolTip(p_name); item->setIcon(QIcon(":/resources/icons/notebook_item.svg")); } void VNotebookSelector::removeNotebookItem(int p_index) { QListWidgetItem *item = m_listWidget->item(p_index + c_notebookStartIdx); m_listWidget->removeItemWidget(item); delete item; } void VNotebookSelector::updateComboBoxItem(int p_index, const QString &p_name) { QListWidgetItem *item = m_listWidget->item(p_index); item->setText(p_name); item->setToolTip(p_name); } void VNotebookSelector::requestPopupListContextMenu(QPoint p_pos) { QModelIndex index = m_listWidget->indexAt(p_pos); if (!index.isValid() || index.row() < c_notebookStartIdx) { return; } QListWidgetItem *item = m_listWidget->itemAt(p_pos); Q_ASSERT(item); m_listWidget->clearSelection(); item->setSelected(true); QMenu menu(this); menu.setToolTipsVisible(true); menu.addAction(m_deleteNotebookAct); menu.addAction(m_notebookInfoAct); menu.exec(m_listWidget->mapToGlobal(p_pos)); } bool VNotebookSelector::eventFilter(QObject *watched, QEvent *event) { QEvent::Type type = event->type(); if (type == QEvent::KeyPress && watched == m_listWidget) { if (handlePopupKeyPress(static_cast(event))) { return true; } } else if (type == QEvent::MouseButtonRelease) { if (static_cast(event)->button() == Qt::RightButton) { return true; } } return QComboBox::eventFilter(watched, event); } int VNotebookSelector::indexOfListItem(const QListWidgetItem *p_item) { int nrItems = m_listWidget->count(); for (int i = 0; i < nrItems; ++i) { if (m_listWidget->item(i) == p_item) { return i; } } return -1; } bool VNotebookSelector::locateNotebook(const VNotebook *p_notebook) { if (p_notebook) { for (int i = 0; i < m_notebooks.size(); ++i) { if (m_notebooks[i] == p_notebook) { setCurrentIndexNotebook(i); return true; } } } return false; } void VNotebookSelector::showPopup() { if (count() <= c_notebookStartIdx) { // No normal notebook items. Just add notebook. newNotebook(); return; } resizeListWidgetToContent(); QComboBox::showPopup(); } void VNotebookSelector::resizeListWidgetToContent() { static QRect screenRect = QGuiApplication::primaryScreen()->geometry(); static int maxMinWidth = screenRect.width() < 400 ? screenRect.width() : screenRect.width() / 2; static int maxMinHeight = screenRect.height() < 400 ? screenRect.height() : screenRect.height() / 2; int minWidth = 0; int minHeight = 0; if (m_listWidget->count() > 0) { // Width minWidth = m_listWidget->sizeHintForColumn(0); minWidth = qMin(minWidth, maxMinWidth); // Height minHeight = m_listWidget->sizeHintForRow(0) * m_listWidget->count() + 10; minHeight = qMin(minHeight, maxMinHeight); } m_listWidget->setMinimumSize(minWidth, minHeight); } void VNotebookSelector::registerNavigation(QChar p_majorKey) { Q_ASSERT(!m_naviLabel); m_majorKey = p_majorKey; } void VNotebookSelector::showNavigation() { if (!isVisible()) { return; } V_ASSERT(!m_naviLabel); m_naviLabel = new QLabel(m_majorKey, this); m_naviLabel->setStyleSheet(g_vnote->getNavigationLabelStyle(m_majorKey)); m_naviLabel->show(); } void VNotebookSelector::hideNavigation() { if (m_naviLabel) { delete m_naviLabel; m_naviLabel = NULL; } } bool VNotebookSelector::handleKeyNavigation(int p_key, bool &p_succeed) { bool ret = false; p_succeed = false; QChar keyChar = VUtils::keyToChar(p_key); if (keyChar == m_majorKey) { // Hit. p_succeed = true; ret = true; if (m_naviLabel) { showPopup(); } } return ret; } bool VNotebookSelector::handlePopupKeyPress(QKeyEvent *p_event) { int key = p_event->key(); int modifiers = p_event->modifiers(); switch (key) { case Qt::Key_BracketLeft: { if (modifiers == Qt::ControlModifier) { p_event->accept(); hidePopup(); return true; } break; } case Qt::Key_J: { if (modifiers == Qt::ControlModifier) { p_event->accept(); QKeyEvent *downEvent = new QKeyEvent(QEvent::KeyPress, Qt::Key_Down, Qt::NoModifier); QCoreApplication::postEvent(m_listWidget, downEvent); return true; } break; } case Qt::Key_K: { if (modifiers == Qt::ControlModifier) { p_event->accept(); QKeyEvent *upEvent = new QKeyEvent(QEvent::KeyPress, Qt::Key_Up, Qt::NoModifier); QCoreApplication::postEvent(m_listWidget, upEvent); return true; } break; } default: break; } return false; }