vnote/src/vdirectorytree.cpp
2018-02-26 19:30:20 +08:00

1212 lines
38 KiB
C++

#include <QtWidgets>
#include "vdirectorytree.h"
#include "dialog/vnewdirdialog.h"
#include "vconfigmanager.h"
#include "dialog/vdirinfodialog.h"
#include "vnote.h"
#include "vdirectory.h"
#include "utils/vutils.h"
#include "veditarea.h"
#include "vconfigmanager.h"
#include "vmainwindow.h"
#include "dialog/vsortdialog.h"
#include "utils/vimnavigationforwidget.h"
#include "utils/viconutils.h"
#include "vfilelist.h"
extern VMainWindow *g_mainWin;
extern VConfigManager *g_config;
extern VNote *g_vnote;
const QString VDirectoryTree::c_infoShortcutSequence = "F2";
const QString VDirectoryTree::c_copyShortcutSequence = "Ctrl+C";
const QString VDirectoryTree::c_cutShortcutSequence = "Ctrl+X";
const QString VDirectoryTree::c_pasteShortcutSequence = "Ctrl+V";
VDirectoryTree::VDirectoryTree(QWidget *parent)
: VTreeWidget(parent),
VNavigationMode(),
m_editArea(NULL)
{
setColumnCount(1);
setHeaderHidden(true);
setContextMenuPolicy(Qt::CustomContextMenu);
initShortcuts();
initActions();
connect(this, SIGNAL(itemExpanded(QTreeWidgetItem*)),
this, SLOT(handleItemExpanded(QTreeWidgetItem*)));
connect(this, SIGNAL(itemCollapsed(QTreeWidgetItem*)),
this, SLOT(handleItemCollapsed(QTreeWidgetItem*)));
connect(this, SIGNAL(customContextMenuRequested(QPoint)),
this, SLOT(contextMenuRequested(QPoint)));
connect(this, &VDirectoryTree::currentItemChanged,
this, &VDirectoryTree::currentDirectoryItemChanged);
}
void VDirectoryTree::initShortcuts()
{
QShortcut *infoShortcut = new QShortcut(QKeySequence(c_infoShortcutSequence), this);
infoShortcut->setContext(Qt::WidgetWithChildrenShortcut);
connect(infoShortcut, &QShortcut::activated,
this, [this](){
editDirectoryInfo();
});
QShortcut *copyShortcut = new QShortcut(QKeySequence(c_copyShortcutSequence), this);
copyShortcut->setContext(Qt::WidgetWithChildrenShortcut);
connect(copyShortcut, &QShortcut::activated,
this, [this](){
copySelectedDirectories();
});
QShortcut *cutShortcut = new QShortcut(QKeySequence(c_cutShortcutSequence), this);
cutShortcut->setContext(Qt::WidgetWithChildrenShortcut);
connect(cutShortcut, &QShortcut::activated,
this, [this](){
cutSelectedDirectories();
});
QShortcut *pasteShortcut = new QShortcut(QKeySequence(c_pasteShortcutSequence), this);
pasteShortcut->setContext(Qt::WidgetWithChildrenShortcut);
connect(pasteShortcut, &QShortcut::activated,
this, [this](){
pasteDirectoriesFromClipboard();
});
QKeySequence seq = QKeySequence(g_config->getShortcutKeySequence("NewSubfolder"));
if (!seq.isEmpty()) {
QShortcut *newSubDirShortcut = new QShortcut(seq, this);
newSubDirShortcut->setContext(Qt::ApplicationShortcut);
connect(newSubDirShortcut, &QShortcut::activated,
this, [this](){
newSubDirectory();
});
}
}
void VDirectoryTree::initActions()
{
m_newNoteAct = new QAction(VIconUtils::menuIcon(":/resources/icons/create_note_tb.svg"),
tr("New &Note"), this);
VUtils::fixTextWithShortcut(m_newNoteAct, "NewNote");
m_newNoteAct->setToolTip(tr("Create a note in selected folder"));
connect(m_newNoteAct, &QAction::triggered,
this, [this]() {
g_mainWin->getFileList()->newFile();
});
newRootDirAct = new QAction(VIconUtils::menuIcon(":/resources/icons/create_rootdir.svg"),
tr("New &Root Folder"), this);
newRootDirAct->setToolTip(tr("Create a root folder in current notebook"));
connect(newRootDirAct, &QAction::triggered,
this, &VDirectoryTree::newRootDirectory);
newSubDirAct = new QAction(VIconUtils::menuIcon(":/resources/icons/create_subdir.svg"),
tr("New &Subfolder"), this);
newSubDirAct->setToolTip(tr("Create a subfolder"));
VUtils::fixTextWithShortcut(newSubDirAct, "NewSubfolder");
connect(newSubDirAct, &QAction::triggered,
this, &VDirectoryTree::newSubDirectory);
deleteDirAct = new QAction(VIconUtils::menuDangerIcon(":/resources/icons/delete_dir.svg"),
tr("&Delete"), this);
deleteDirAct->setToolTip(tr("Delete selected folder"));
connect(deleteDirAct, &QAction::triggered,
this, &VDirectoryTree::deleteSelectedDirectory);
dirInfoAct = new QAction(VIconUtils::menuIcon(":/resources/icons/dir_info.svg"),
tr("&Info\t%1").arg(VUtils::getShortcutText(c_infoShortcutSequence)), this);
dirInfoAct->setToolTip(tr("View and edit current folder's information"));
connect(dirInfoAct, &QAction::triggered,
this, &VDirectoryTree::editDirectoryInfo);
copyAct = new QAction(VIconUtils::menuIcon(":/resources/icons/copy.svg"),
tr("&Copy\t%1").arg(VUtils::getShortcutText(c_copyShortcutSequence)), this);
copyAct->setToolTip(tr("Copy selected folders"));
connect(copyAct, &QAction::triggered,
this, &VDirectoryTree::copySelectedDirectories);
cutAct = new QAction(VIconUtils::menuIcon(":/resources/icons/cut.svg"),
tr("C&ut\t%1").arg(VUtils::getShortcutText(c_cutShortcutSequence)), this);
cutAct->setToolTip(tr("Cut selected folders"));
connect(cutAct, &QAction::triggered,
this, &VDirectoryTree::cutSelectedDirectories);
pasteAct = new QAction(VIconUtils::menuIcon(":/resources/icons/paste.svg"),
tr("&Paste\t%1").arg(VUtils::getShortcutText(c_pasteShortcutSequence)), this);
pasteAct->setToolTip(tr("Paste folders in this folder"));
connect(pasteAct, &QAction::triggered,
this, &VDirectoryTree::pasteDirectoriesFromClipboard);
m_openLocationAct = new QAction(tr("&Open Folder Location"), this);
m_openLocationAct->setToolTip(tr("Open the folder containing this folder in operating system"));
connect(m_openLocationAct, &QAction::triggered,
this, &VDirectoryTree::openDirectoryLocation);
m_reloadAct = new QAction(tr("&Reload From Disk"), this);
m_reloadAct->setToolTip(tr("Reload the content of this folder (or notebook) from disk"));
connect(m_reloadAct, &QAction::triggered,
this, &VDirectoryTree::reloadFromDisk);
m_sortAct = new QAction(VIconUtils::menuIcon(":/resources/icons/sort.svg"),
tr("&Sort"),
this);
m_sortAct->setToolTip(tr("Sort folders in this folder/notebook manually"));
connect(m_sortAct, SIGNAL(triggered(bool)),
this, SLOT(sortItems()));
}
void VDirectoryTree::setNotebook(VNotebook *p_notebook)
{
if (m_notebook == p_notebook) {
return;
}
clear();
m_notebook = p_notebook;
if (!m_notebook) {
return;
}
if (!m_notebook->open()) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to open notebook <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(m_notebook->getName()),
tr("Please check if the notebook's root folder <span style=\"%1\">%2</span> exists.")
.arg(g_config->c_dataTextStyle)
.arg(m_notebook->getPath()),
QMessageBox::Ok,
QMessageBox::Ok,
this);
return;
}
updateDirectoryTree();
}
void VDirectoryTree::fillTreeItem(QTreeWidgetItem *p_item, VDirectory *p_directory)
{
int col = 0;
QString name = p_directory->getName();
p_item->setText(col, name);
p_item->setToolTip(col, name);
p_item->setData(col, Qt::UserRole, QVariant::fromValue(p_directory));
p_item->setIcon(col, VIconUtils::treeViewIcon(":/resources/icons/dir_item.svg"));
}
void VDirectoryTree::updateDirectoryTree()
{
clear();
VDirectory *rootDir = m_notebook->getRootDir();
const QVector<VDirectory *> &subDirs = rootDir->getSubDirs();
for (int i = 0; i < subDirs.size(); ++i) {
VDirectory *dir = subDirs[i];
QTreeWidgetItem *item = new QTreeWidgetItem(this);
fillTreeItem(item, dir);
buildSubTree(item, 1);
}
if (!restoreCurrentItem() && topLevelItemCount() > 0) {
setCurrentItem(topLevelItem(0));
}
}
bool VDirectoryTree::restoreCurrentItem()
{
auto it = m_notebookCurrentDirMap.find(m_notebook);
if (it != m_notebookCurrentDirMap.end()) {
QTreeWidgetItem *item = findVDirectory(it.value());
if (item) {
setCurrentItem(item);
return true;
}
}
return false;
}
void VDirectoryTree::buildSubTree(QTreeWidgetItem *p_parent, int p_depth)
{
if (p_depth == 0) {
return;
}
Q_ASSERT(p_parent);
VDirectory *dir = getVDirectory(p_parent);
if (!dir->open()) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to open folder <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(dir->getName()),
tr("Please check if directory <span style=\"%1\">%2</span> exists.")
.arg(g_config->c_dataTextStyle)
.arg(dir->fetchPath()),
QMessageBox::Ok,
QMessageBox::Ok,
this);
return;
}
if (p_parent->childCount() > 0) {
// This directory has been built before. Try its children directly.
int cnt = p_parent->childCount();
for (int i = 0; i < cnt; ++i) {
buildSubTree(p_parent->child(i), p_depth -1);
}
} else {
const QVector<VDirectory *> &subDirs = dir->getSubDirs();
for (int i = 0; i < subDirs.size(); ++i) {
VDirectory *subDir = subDirs[i];
QTreeWidgetItem *item = new QTreeWidgetItem(p_parent);
fillTreeItem(item, subDir);
buildSubTree(item, p_depth - 1);
}
}
if (dir->isExpanded()) {
expandItem(p_parent);
}
}
void VDirectoryTree::handleItemCollapsed(QTreeWidgetItem *p_item)
{
if (p_item) {
VDirectory *dir = getVDirectory(p_item);
dir->setExpanded(false);
}
}
void VDirectoryTree::handleItemExpanded(QTreeWidgetItem *p_item)
{
if (p_item) {
buildChildren(p_item);
VDirectory *dir = getVDirectory(p_item);
dir->setExpanded(true);
}
}
void VDirectoryTree::buildChildren(QTreeWidgetItem *p_item)
{
Q_ASSERT(p_item);
int nrChild = p_item->childCount();
if (nrChild == 0) {
return;
}
for (int i = 0; i < nrChild; ++i) {
QTreeWidgetItem *childItem = p_item->child(i);
if (childItem->childCount() > 0) {
continue;
}
buildSubTree(childItem, 1);
}
}
void VDirectoryTree::updateItemDirectChildren(QTreeWidgetItem *p_item)
{
QPointer<VDirectory> parentDir;
if (p_item) {
parentDir = getVDirectory(p_item);
} else {
parentDir = m_notebook->getRootDir();
}
const QVector<VDirectory *> &dirs = parentDir->getSubDirs();
QHash<VDirectory *, QTreeWidgetItem *> itemDirMap;
int nrChild = p_item ? p_item->childCount() : topLevelItemCount();
for (int i = 0; i < nrChild; ++i) {
QTreeWidgetItem *item = p_item ? p_item->child(i) : topLevelItem(i);
itemDirMap.insert(getVDirectory(item), item);
}
for (int i = 0; i < dirs.size(); ++i) {
VDirectory *dir = dirs[i];
QTreeWidgetItem *item = itemDirMap.value(dir, NULL);
if (item) {
if (p_item) {
p_item->removeChild(item);
p_item->insertChild(i, item);
} else {
int topIdx = indexOfTopLevelItem(item);
takeTopLevelItem(topIdx);
insertTopLevelItem(i, item);
}
itemDirMap.remove(dir);
} else {
// Insert a new item
if (p_item) {
item = new QTreeWidgetItem(p_item);
} else {
item = new QTreeWidgetItem(this);
}
fillTreeItem(item, dir);
buildSubTree(item, 1);
}
expandSubTree(item);
}
// Delete items without corresponding VDirectory
for (auto iter = itemDirMap.begin(); iter != itemDirMap.end(); ++iter) {
QTreeWidgetItem *item = iter.value();
if (p_item) {
p_item->removeChild(item);
} else {
int topIdx = indexOfTopLevelItem(item);
takeTopLevelItem(topIdx);
}
delete item;
}
}
void VDirectoryTree::contextMenuRequested(QPoint pos)
{
QTreeWidgetItem *item = itemAt(pos);
if (!m_notebook) {
return;
}
QMenu menu(this);
menu.setToolTipsVisible(true);
if (!item) {
// Context menu on the free space of the QTreeWidget
menu.addAction(newRootDirAct);
if (topLevelItemCount() > 1) {
menu.addAction(m_sortAct);
}
} else {
// Context menu on a QTreeWidgetItem
menu.addAction(m_newNoteAct);
if (item->parent()) {
// Low-level item
menu.addAction(newSubDirAct);
if (item->parent()->childCount() > 1) {
menu.addAction(m_sortAct);
}
} else {
// Top-level item
menu.addAction(newRootDirAct);
menu.addAction(newSubDirAct);
if (topLevelItemCount() > 1) {
menu.addAction(m_sortAct);
}
}
menu.addSeparator();
menu.addAction(deleteDirAct);
menu.addAction(copyAct);
menu.addAction(cutAct);
}
if (pasteAvailable()) {
if (!item) {
menu.addSeparator();
}
menu.addAction(pasteAct);
}
menu.addSeparator();
menu.addAction(m_reloadAct);
if (item) {
menu.addAction(m_openLocationAct);
menu.addAction(dirInfoAct);
}
menu.exec(mapToGlobal(pos));
}
void VDirectoryTree::newSubDirectory()
{
if (!m_notebook) {
return;
}
QTreeWidgetItem *curItem = currentItem();
if (!curItem) {
return;
}
VDirectory *curDir = getVDirectory(curItem);
QString info = tr("Create a subfolder in <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(curDir->getName());
QString defaultName("new_folder");
defaultName = VUtils::getDirNameWithSequence(curDir->fetchPath(), defaultName);
VNewDirDialog dialog(tr("Create Folder"), info, defaultName, curDir, this);
if (dialog.exec() == QDialog::Accepted) {
QString name = dialog.getNameInput();
QString msg;
VDirectory *subDir = curDir->createSubDirectory(name, &msg);
if (!subDir) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to create subfolder <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(name),
msg,
QMessageBox::Ok,
QMessageBox::Ok,
this);
return;
}
updateItemDirectChildren(curItem);
locateDirectory(subDir);
}
}
void VDirectoryTree::newRootDirectory()
{
if (!m_notebook) {
return;
}
VDirectory *rootDir = m_notebook->getRootDir();
QString info = tr("Create a root folder in notebook <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(m_notebook->getName());
QString defaultName("new_folder");
defaultName = VUtils::getDirNameWithSequence(rootDir->fetchPath(), defaultName);
VNewDirDialog dialog(tr("Create Root Folder"), info, defaultName, rootDir, this);
if (dialog.exec() == QDialog::Accepted) {
QString name = dialog.getNameInput();
QString msg;
VDirectory *dir = rootDir->createSubDirectory(name, &msg);
if (!dir) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to create root folder <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(name),
msg,
QMessageBox::Ok,
QMessageBox::Ok,
this);
return;
}
updateItemDirectChildren(NULL);
locateDirectory(dir);
}
}
void VDirectoryTree::deleteSelectedDirectory()
{
Q_ASSERT(selectedItems().size() <= 1);
QTreeWidgetItem *curItem = currentItem();
if (!curItem) {
return;
}
VDirectory *curDir = getVDirectory(curItem);
int ret = VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Are you sure to delete folder <span style=\"%1\">%2</span>?")
.arg(g_config->c_dataTextStyle)
.arg(curDir->getName()),
tr("<span style=\"%1\">WARNING</span>: "
"VNote will delete the whole directory "
"<span style=\"%2\">%3</span>."
"Deleted files could be found in the recycle bin "
"of this folder.<br>"
"The operation is IRREVERSIBLE!")
.arg(g_config->c_warningTextStyle)
.arg(g_config->c_dataTextStyle)
.arg(curDir->fetchPath()),
QMessageBox::Ok | QMessageBox::Cancel,
QMessageBox::Ok,
this,
MessageBoxType::Danger);
if (ret == QMessageBox::Ok) {
int nrDeleted = 1;
m_editArea->closeFile(curDir, true);
// Remove the item from the tree.
delete curItem;
QString msg;
QString dirName = curDir->getName();
QString dirPath = curDir->fetchPath();
if (!VDirectory::deleteDirectory(curDir, false, &msg)) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to delete folder <span style=\"%1\">%2</span>.<br>"
"Please check <span style=\"%1\">%3</span> and manually delete it.")
.arg(g_config->c_dataTextStyle)
.arg(dirName)
.arg(dirPath),
msg,
QMessageBox::Ok,
QMessageBox::Ok,
this);
} else {
g_mainWin->showStatusMessage(tr("%1 %2 deleted")
.arg(nrDeleted)
.arg(nrDeleted > 1 ? tr("folders") : tr("folder")));
}
}
}
void VDirectoryTree::currentDirectoryItemChanged(QTreeWidgetItem *currentItem)
{
if (!currentItem) {
emit currentDirectoryChanged(NULL);
return;
}
QPointer<VDirectory> dir = getVDirectory(currentItem);
m_notebookCurrentDirMap[m_notebook] = dir;
emit currentDirectoryChanged(dir);
}
void VDirectoryTree::editDirectoryInfo()
{
QTreeWidgetItem *curItem = currentItem();
if (!curItem) {
return;
}
VDirectory *curDir = getVDirectory(curItem);
QString curName = curDir->getName();
VDirInfoDialog dialog(tr("Folder Information"),
"",
curDir,
curDir->getParentDirectory(),
this);
if (dialog.exec() == QDialog::Accepted) {
QString name = dialog.getNameInput();
if (name == curName) {
return;
}
if (!curDir->rename(name)) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to rename folder <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(curName),
"",
QMessageBox::Ok,
QMessageBox::Ok,
this);
return;
}
fillTreeItem(curItem, curDir);
emit directoryUpdated(curDir, UpdateAction::InfoChanged);
}
}
void VDirectoryTree::openDirectoryLocation() const
{
QTreeWidgetItem *curItem = currentItem();
V_ASSERT(curItem);
QUrl url = QUrl::fromLocalFile(getVDirectory(curItem)->fetchBasePath());
QDesktopServices::openUrl(url);
}
void VDirectoryTree::reloadFromDisk()
{
if (!m_notebook) {
return;
}
QString msg;
QString info;
VDirectory *curDir = NULL;
QTreeWidgetItem *curItem = currentItem();
if (curItem) {
// Reload current directory.
curDir = getVDirectory(curItem);
info = tr("Are you sure to reload folder <span style=\"%1\">%2</span>?")
.arg(g_config->c_dataTextStyle).arg(curDir->getName());
msg = tr("Folder %1 reloaded from disk").arg(curDir->getName());
} else {
// Reload notebook.
info = tr("Are you sure to reload notebook <span style=\"%1\">%2</span>?")
.arg(g_config->c_dataTextStyle).arg(m_notebook->getName());
msg = tr("Notebook %1 reloaded from disk").arg(m_notebook->getName());
}
if (g_config->getConfirmReloadFolder()) {
int ret = VUtils::showMessage(QMessageBox::Information, tr("Information"),
info,
tr("VNote will close all the related notes before reload."),
QMessageBox::Ok | QMessageBox::YesToAll | QMessageBox::Cancel,
QMessageBox::Ok,
this);
switch (ret) {
case QMessageBox::YesToAll:
g_config->setConfirmReloadFolder(false);
// Fall through.
case QMessageBox::Ok:
break;
case QMessageBox::Cancel:
return;
default:
return;
}
}
m_notebookCurrentDirMap.remove(m_notebook);
if (curItem) {
if (!m_editArea->closeFile(curDir, false)) {
return;
}
setCurrentItem(NULL);
curItem->setExpanded(false);
curDir->setExpanded(false);
curDir->close();
// Remove all its children.
QList<QTreeWidgetItem *> children = curItem->takeChildren();
for (int i = 0; i < children.size(); ++i) {
delete children[i];
}
buildSubTree(curItem, 1);
setCurrentItem(curItem);
} else {
if (!m_editArea->closeFile(m_notebook, false)) {
return;
}
m_notebook->close();
if (!m_notebook->open()) {
VUtils::showMessage(QMessageBox::Warning, tr("Warning"),
tr("Fail to open notebook <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle).arg(m_notebook->getName()),
tr("Please check if path <span style=\"%1\">%2</span> exists.")
.arg(g_config->c_dataTextStyle).arg(m_notebook->getPath()),
QMessageBox::Ok, QMessageBox::Ok, this);
clear();
return;
}
updateDirectoryTree();
}
if (!msg.isEmpty()) {
g_mainWin->showStatusMessage(msg);
}
}
void VDirectoryTree::copySelectedDirectories(bool p_isCut)
{
QList<QTreeWidgetItem *> items = selectedItems();
if (items.isEmpty()) {
return;
}
QJsonArray dirs;
for (int i = 0; i < items.size(); ++i) {
VDirectory *dir = getVDirectory(items[i]);
dirs.append(dir->fetchPath());
}
QJsonObject clip;
clip[ClipboardConfig::c_magic] = getNewMagic();
clip[ClipboardConfig::c_type] = (int)ClipboardOpType::CopyDir;
clip[ClipboardConfig::c_isCut] = p_isCut;
clip[ClipboardConfig::c_dirs] = dirs;
QClipboard *clipboard = QApplication::clipboard();
clipboard->setText(QJsonDocument(clip).toJson(QJsonDocument::Compact));
qDebug() << "copied directories info" << clipboard->text();
int cnt = dirs.size();
g_mainWin->showStatusMessage(tr("%1 %2 %3")
.arg(cnt)
.arg(cnt > 1 ? tr("folders") : tr("folder"))
.arg(p_isCut ? tr("cut") : tr("copied")));
}
void VDirectoryTree::cutSelectedDirectories()
{
copySelectedDirectories(true);
}
void VDirectoryTree::pasteDirectoriesFromClipboard()
{
if (!pasteAvailable()) {
return;
}
QJsonObject obj = VUtils::clipboardToJson();
QJsonArray dirs = obj[ClipboardConfig::c_dirs].toArray();
bool isCut = obj[ClipboardConfig::c_isCut].toBool();
QVector<QString> dirsToPaste(dirs.size());
for (int i = 0; i < dirs.size(); ++i) {
dirsToPaste[i] = dirs[i].toString();
}
VDirectory *destDir;
QTreeWidgetItem *item = currentItem();
if (item) {
destDir = getVDirectory(item);
} else {
destDir = m_notebook->getRootDir();
}
pasteDirectories(destDir, dirsToPaste, isCut);
QClipboard *clipboard = QApplication::clipboard();
clipboard->clear();
}
void VDirectoryTree::pasteDirectories(VDirectory *p_destDir,
const QVector<QString> &p_dirs,
bool p_isCut)
{
if (!p_destDir || p_dirs.isEmpty()) {
return;
}
int nrPasted = 0;
for (int i = 0; i < p_dirs.size(); ++i) {
VDirectory *dir = g_vnote->getInternalDirectory(p_dirs[i]);
if (!dir) {
qWarning() << "Copied dir is not an internal folder" << p_dirs[i];
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to paste folder <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(p_dirs[i]),
tr("VNote could not find this folder in any notebook."),
QMessageBox::Ok,
QMessageBox::Ok,
this);
continue;
}
if (dir == p_destDir) {
continue;
}
QString dirName = dir->getName();
VDirectory *paDir = dir->getParentDirectory();
if (paDir == p_destDir) {
if (p_isCut) {
continue;
}
// Copy and paste in the same folder.
// Rename it to xxx_copy.
dirName = VUtils::generateCopiedDirName(paDir->fetchPath(), dirName);
} else {
// Rename it to xxx_copy if needed.
dirName = VUtils::generateCopiedDirName(p_destDir->fetchPath(), dirName);
}
QString msg;
VDirectory *destDir = NULL;
bool ret = VDirectory::copyDirectory(p_destDir,
dirName,
dir,
p_isCut,
&destDir,
&msg);
if (!ret) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to copy folder <span style=\"%1\">%2</span>.")
.arg(g_config->c_dataTextStyle)
.arg(p_dirs[i]),
msg,
QMessageBox::Ok,
QMessageBox::Ok,
this);
}
if (destDir) {
++nrPasted;
// Update QTreeWidget.
bool isWidget;
QTreeWidgetItem *destItem = findVDirectory(p_destDir, &isWidget);
if (destItem || isWidget) {
updateItemDirectChildren(destItem);
}
if (p_isCut) {
QTreeWidgetItem *srcItem = findVDirectory(paDir, &isWidget);
if (srcItem || isWidget) {
updateItemDirectChildren(srcItem);
}
}
// Broadcast this update
emit directoryUpdated(destDir, p_isCut ? UpdateAction::Moved : UpdateAction::InfoChanged);
}
}
qDebug() << "pasted" << nrPasted << "directories";
if (nrPasted > 0) {
g_mainWin->showStatusMessage(tr("%1 %2 pasted")
.arg(nrPasted)
.arg(nrPasted > 1 ? tr("folders") : tr("folder")));
}
getNewMagic();
}
bool VDirectoryTree::pasteAvailable() const
{
QJsonObject obj = VUtils::clipboardToJson();
if (obj.isEmpty()) {
return false;
}
if (!obj.contains(ClipboardConfig::c_type)) {
return false;
}
ClipboardOpType type = (ClipboardOpType)obj[ClipboardConfig::c_type].toInt();
if (type != ClipboardOpType::CopyDir) {
return false;
}
if (!obj.contains(ClipboardConfig::c_magic)
|| !obj.contains(ClipboardConfig::c_isCut)
|| !obj.contains(ClipboardConfig::c_dirs)) {
return false;
}
int magic = obj[ClipboardConfig::c_magic].toInt();
if (!checkMagic(magic)) {
return false;
}
QJsonArray dirs = obj[ClipboardConfig::c_dirs].toArray();
return !dirs.isEmpty();
}
void VDirectoryTree::mousePressEvent(QMouseEvent *event)
{
QTreeWidgetItem *item = itemAt(event->pos());
if (!item) {
setCurrentItem(NULL);
}
VTreeWidget::mousePressEvent(event);
}
void VDirectoryTree::keyPressEvent(QKeyEvent *event)
{
int key = event->key();
int modifiers = event->modifiers();
switch (key) {
case Qt::Key_Return:
{
QTreeWidgetItem *item = currentItem();
if (item) {
item->setExpanded(!item->isExpanded());
}
break;
}
case Qt::Key_Asterisk:
{
if (modifiers == Qt::ShiftModifier) {
// *, by default will expand current item recursively.
// We build the tree recursively before the expanding.
QTreeWidgetItem *item = currentItem();
if (item) {
buildSubTree(item, -1);
}
}
break;
}
default:
break;
}
VTreeWidget::keyPressEvent(event);
}
QTreeWidgetItem *VDirectoryTree::findVDirectory(const VDirectory *p_dir, bool *p_widget)
{
if (p_widget) {
*p_widget = false;
}
if (!p_dir) {
return NULL;
} else if (p_dir->getNotebookName() != m_notebook->getName()) {
return NULL;
} else if (p_dir == m_notebook->getRootDir()) {
if (p_widget) {
*p_widget = true;
}
return NULL;
}
bool isWidget;
QTreeWidgetItem *pItem = findVDirectory(p_dir->getParentDirectory(), &isWidget);
if (pItem) {
// Iterate all its children to find the match.
int nrChild = pItem->childCount();
for (int i = 0; i < nrChild; ++i) {
QTreeWidgetItem *item = pItem->child(i);
if (getVDirectory(item) == p_dir) {
return item;
}
}
} else if (isWidget) {
// Iterate all the top-level items.
int nrChild = topLevelItemCount();
for (int i = 0; i < nrChild; ++i) {
QTreeWidgetItem *item = topLevelItem(i);
if (getVDirectory(item) == p_dir) {
return item;
}
}
}
return NULL;
}
bool VDirectoryTree::locateDirectory(const VDirectory *p_directory)
{
if (p_directory) {
if (p_directory->getNotebook() != m_notebook) {
return false;
}
QTreeWidgetItem *item = expandToVDirectory(p_directory);
if (item) {
setCurrentItem(item);
}
return item;
}
return false;
}
QTreeWidgetItem *VDirectoryTree::expandToVDirectory(const VDirectory *p_directory)
{
if (!p_directory
|| p_directory->getNotebook() != m_notebook
|| p_directory == m_notebook->getRootDir()) {
return NULL;
}
if (p_directory->getParentDirectory() == m_notebook->getRootDir()) {
// Top-level items.
int nrChild = topLevelItemCount();
for (int i = 0; i < nrChild; ++i) {
QTreeWidgetItem *item = topLevelItem(i);
if (getVDirectory(item) == p_directory) {
return item;
}
}
} else {
QTreeWidgetItem *pItem = expandToVDirectory(p_directory->getParentDirectory());
if (!pItem) {
return NULL;
}
int nrChild = pItem->childCount();
if (nrChild == 0) {
buildSubTree(pItem, 1);
}
nrChild = pItem->childCount();
for (int i = 0; i < nrChild; ++i) {
QTreeWidgetItem *item = pItem->child(i);
if (getVDirectory(item) == p_directory) {
return item;
}
}
}
return NULL;
}
void VDirectoryTree::expandSubTree(QTreeWidgetItem *p_item)
{
if (!p_item) {
return;
}
VDirectory *dir = getVDirectory(p_item);
int nrChild = p_item->childCount();
for (int i = 0; i < nrChild; ++i) {
expandSubTree(p_item->child(i));
}
if (dir->isExpanded()) {
Q_ASSERT(nrChild > 0);
expandItem(p_item);
}
}
void VDirectoryTree::showNavigation()
{
VNavigationMode::showNavigation(this);
}
bool VDirectoryTree::handleKeyNavigation(int p_key, bool &p_succeed)
{
static bool secondKey = false;
return VNavigationMode::handleKeyNavigation(this,
secondKey,
p_key,
p_succeed);
}
int VDirectoryTree::getNewMagic()
{
m_magicForClipboard = (int)QDateTime::currentDateTime().toTime_t();
m_magicForClipboard |= qrand();
return m_magicForClipboard;
}
bool VDirectoryTree::checkMagic(int p_magic) const
{
return m_magicForClipboard == p_magic;
}
void VDirectoryTree::sortItems()
{
if (!m_notebook) {
return;
}
QTreeWidgetItem *item = currentItem();
if (item && item->parent()) {
sortItems(getVDirectory(item->parent()));
} else {
sortItems(m_notebook->getRootDir());
}
if (item) {
setCurrentItem(item);
}
}
void VDirectoryTree::sortItems(VDirectory *p_dir)
{
if (!p_dir) {
return;
}
const QVector<VDirectory *> &dirs = p_dir->getSubDirs();
if (dirs.size() < 2) {
return;
}
bool isNotebook = p_dir->parent() == NULL;
VSortDialog dialog(tr("Sort Folders"),
tr("Sort folders in %1 <span style=\"%2\">%3</span> "
"in the configuration file.")
.arg(isNotebook ? tr("notebook") : tr("folder"))
.arg(g_config->c_dataTextStyle)
.arg(isNotebook ? p_dir->getNotebook()->getName() : p_dir->getName()),
this);
QTreeWidget *tree = dialog.getTreeWidget();
tree->clear();
tree->setColumnCount(2);
QStringList headers;
headers << tr("Name") << tr("Created Time");
tree->setHeaderLabels(headers);
for (int i = 0; i < dirs.size(); ++i) {
const VDirectory *dir = dirs[i];
Q_ASSERT(dir->isOpened());
QString createdTime = VUtils::displayDateTime(dir->getCreatedTimeUtc().toLocalTime());
QStringList cols;
cols << dir->getName() << createdTime;
QTreeWidgetItem *item = new QTreeWidgetItem(tree, cols);
item->setData(0, Qt::UserRole, i);
}
dialog.treeUpdated();
if (dialog.exec()) {
QVector<QVariant> data = dialog.getSortedData();
Q_ASSERT(data.size() == dirs.size());
QVector<int> sortedIdx(data.size(), -1);
for (int i = 0; i < data.size(); ++i) {
sortedIdx[i] = data[i].toInt();
}
qDebug() << "sort dirs" << sortedIdx;
if (!p_dir->sortSubDirectories(sortedIdx)) {
VUtils::showMessage(QMessageBox::Warning,
tr("Warning"),
tr("Fail to sort folders in %1 <span style=\"%2\">%3</span>.")
.arg(isNotebook ? tr("notebook") : tr("folder"))
.arg(g_config->c_dataTextStyle)
.arg(isNotebook ? p_dir->getNotebook()->getName() : p_dir->getName()),
"",
QMessageBox::Ok,
QMessageBox::Ok,
this);
}
updateItemDirectChildren(findVDirectory(p_dir));
}
}
VDirectory *VDirectoryTree::currentDirectory() const
{
QTreeWidgetItem *item = currentItem();
if (item) {
return getVDirectory(item);
}
return NULL;
}