mirror of
https://gitee.com/vnotex/vnote.git
synced 2025-07-05 13:59:52 +08:00
515 lines
15 KiB
C++
515 lines
15 KiB
C++
#include "vnotebookselector.h"
|
|
#include <QDebug>
|
|
#include <QJsonObject>
|
|
#include <QListWidget>
|
|
#include <QAction>
|
|
#include <QMenu>
|
|
#include <QGuiApplication>
|
|
#include <QScreen>
|
|
#include <QLabel>
|
|
#include <QDesktopServices>
|
|
#include <QUrl>
|
|
#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 "
|
|
"<span style=\"%1\">%2</span> in <span style=\"%1\">%3</span>.")
|
|
.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<QListWidgetItem *> 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 "
|
|
"<span style=\"%1\">%2</span> 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<QListWidgetItem *> 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<QKeyEvent *>(event))) {
|
|
return true;
|
|
}
|
|
} else if (type == QEvent::MouseButtonRelease) {
|
|
if (static_cast<QMouseEvent *>(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;
|
|
}
|