mirror of
https://gitee.com/vnotex/vnote.git
synced 2025-07-05 13:59:52 +08:00
989 lines
34 KiB
C++
989 lines
34 KiB
C++
#include "exportdialog.h"
|
|
|
|
#include <QGroupBox>
|
|
#include <QFormLayout>
|
|
#include <QVBoxLayout>
|
|
#include <QHBoxLayout>
|
|
#include <QLabel>
|
|
#include <QPushButton>
|
|
#include <QCheckBox>
|
|
#include <QLineEdit>
|
|
#include <QProgressBar>
|
|
#include <QFileInfo>
|
|
#include <QFileDialog>
|
|
#include <QUrl>
|
|
#include <QPlainTextEdit>
|
|
#include <QCoreApplication>
|
|
#include <QPrinter>
|
|
#include <QPageSetupDialog>
|
|
#include <QPageLayout>
|
|
#include <QInputDialog>
|
|
|
|
#include <notebook/notebook.h>
|
|
#include <notebook/node.h>
|
|
#include <buffer/buffer.h>
|
|
#include <widgets/widgetsfactory.h>
|
|
#include <core/thememgr.h>
|
|
#include <core/configmgr.h>
|
|
#include <core/sessionconfig.h>
|
|
#include <core/vnotex.h>
|
|
#include <utils/widgetutils.h>
|
|
#include <utils/fileutils.h>
|
|
#include <utils/pathutils.h>
|
|
#include <utils/clipboardutils.h>
|
|
#include <export/exporter.h>
|
|
#include <widgets/locationinputwithbrowsebutton.h>
|
|
#include <widgets/messageboxhelper.h>
|
|
#include <widgets/combobox.h>
|
|
|
|
using namespace vnotex;
|
|
|
|
ExportDialog::ExportDialog(Notebook *p_notebook,
|
|
Node *p_folder,
|
|
Node *p_note,
|
|
Buffer *p_buffer,
|
|
QWidget *p_parent)
|
|
: ScrollDialog(p_parent),
|
|
m_notebook(p_notebook),
|
|
m_folder(p_folder),
|
|
m_note(p_note),
|
|
m_buffer(p_buffer)
|
|
{
|
|
setupUI();
|
|
|
|
initOptions();
|
|
|
|
restoreFields(m_option);
|
|
|
|
setWindowFlags(windowFlags() | Qt::WindowMinimizeButtonHint);
|
|
|
|
connect(this, &QDialog::finished,
|
|
this, [this]() {
|
|
saveFields(m_option);
|
|
ConfigMgr::getInst().getSessionConfig().setExportOption(m_option);
|
|
});
|
|
}
|
|
|
|
void ExportDialog::setupUI()
|
|
{
|
|
auto widget = new QWidget(this);
|
|
setCentralWidget(widget);
|
|
|
|
auto mainLayout = new QVBoxLayout(widget);
|
|
|
|
auto sourceBox = setupSourceGroup(widget);
|
|
mainLayout->addWidget(sourceBox);
|
|
|
|
auto targetBox = setupTargetGroup(widget);
|
|
mainLayout->addWidget(targetBox);
|
|
|
|
m_advancedGroupBox = setupAdvancedGroup(widget);
|
|
mainLayout->addWidget(m_advancedGroupBox);
|
|
|
|
m_progressBar = new QProgressBar(widget);
|
|
m_progressBar->setRange(0, 0);
|
|
m_progressBar->hide();
|
|
addBottomWidget(m_progressBar);
|
|
|
|
setupButtonBox();
|
|
|
|
setWindowTitle(tr("Export"));
|
|
}
|
|
|
|
QGroupBox *ExportDialog::setupSourceGroup(QWidget *p_parent)
|
|
{
|
|
auto box = new QGroupBox(tr("Source"), p_parent);
|
|
auto layout = WidgetsFactory::createFormLayout(box);
|
|
|
|
{
|
|
m_sourceComboBox = WidgetsFactory::createComboBox(box);
|
|
if (m_buffer) {
|
|
m_sourceComboBox->addItem(tr("Current Buffer (%1)").arg(m_buffer->getName()),
|
|
static_cast<int>(ExportSource::CurrentBuffer));
|
|
}
|
|
if (m_note && m_note->hasContent()) {
|
|
m_sourceComboBox->addItem(tr("Current Note (%1)").arg(m_note->getName()),
|
|
static_cast<int>(ExportSource::CurrentNote));
|
|
}
|
|
if (m_folder && m_folder->isContainer()) {
|
|
m_sourceComboBox->addItem(tr("Current Folder (%1)").arg(m_folder->getName()),
|
|
static_cast<int>(ExportSource::CurrentFolder));
|
|
}
|
|
if (m_notebook) {
|
|
m_sourceComboBox->addItem(tr("Current Notebook (%1)").arg(m_notebook->getName()),
|
|
static_cast<int>(ExportSource::CurrentNotebook));
|
|
}
|
|
layout->addRow(tr("Source:"), m_sourceComboBox);
|
|
}
|
|
|
|
{
|
|
// TODO: Source format filtering.
|
|
}
|
|
|
|
return box;
|
|
}
|
|
|
|
QString ExportDialog::getDefaultOutputDir() const
|
|
{
|
|
return PathUtils::concatenateFilePath(ConfigMgr::getDocumentOrHomePath(), tr("vnote_exports"));
|
|
}
|
|
|
|
QGroupBox *ExportDialog::setupTargetGroup(QWidget *p_parent)
|
|
{
|
|
auto box = new QGroupBox(tr("Target"), p_parent);
|
|
auto layout = WidgetsFactory::createFormLayout(box);
|
|
|
|
{
|
|
m_targetFormatComboBox = WidgetsFactory::createComboBox(box);
|
|
m_targetFormatComboBox->addItem(tr("Markdown"),
|
|
static_cast<int>(ExportFormat::Markdown));
|
|
m_targetFormatComboBox->addItem(tr("HTML"),
|
|
static_cast<int>(ExportFormat::HTML));
|
|
m_targetFormatComboBox->addItem(tr("PDF"),
|
|
static_cast<int>(ExportFormat::PDF));
|
|
m_targetFormatComboBox->addItem(tr("Custom"),
|
|
static_cast<int>(ExportFormat::Custom));
|
|
connect(m_targetFormatComboBox, QOverload<int>::of(&QComboBox::currentIndexChanged),
|
|
this, [this]() {
|
|
AdvancedSettings settings = AdvancedSettings::Max;
|
|
int format = m_targetFormatComboBox->currentData().toInt();
|
|
switch (format) {
|
|
case static_cast<int>(ExportFormat::HTML):
|
|
settings = AdvancedSettings::HTML;
|
|
break;
|
|
|
|
case static_cast<int>(ExportFormat::PDF):
|
|
settings = AdvancedSettings::PDF;
|
|
break;
|
|
|
|
case static_cast<int>(ExportFormat::Custom):
|
|
settings = AdvancedSettings::Custom;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
showAdvancedSettings(settings);
|
|
});
|
|
layout->addRow(tr("Format:"), m_targetFormatComboBox);
|
|
}
|
|
|
|
{
|
|
m_transparentBgCheckBox = WidgetsFactory::createCheckBox(tr("Use transparent background"), box);
|
|
layout->addRow(m_transparentBgCheckBox);
|
|
}
|
|
|
|
{
|
|
const auto webStyles = VNoteX::getInst().getThemeMgr().getWebStyles();
|
|
|
|
m_renderingStyleComboBox = WidgetsFactory::createComboBox(box);
|
|
layout->addRow(tr("Rendering style:"), m_renderingStyleComboBox);
|
|
for (const auto &pa : webStyles) {
|
|
m_renderingStyleComboBox->addItem(pa.first, pa.second);
|
|
}
|
|
|
|
m_syntaxHighlightStyleComboBox = WidgetsFactory::createComboBox(box);
|
|
layout->addRow(tr("Syntax highlighting style:"), m_syntaxHighlightStyleComboBox);
|
|
for (const auto &pa : webStyles) {
|
|
m_syntaxHighlightStyleComboBox->addItem(pa.first, pa.second);
|
|
}
|
|
}
|
|
|
|
{
|
|
m_outputDirInput = new LocationInputWithBrowseButton(box);
|
|
layout->addRow(tr("Output directory:"), m_outputDirInput);
|
|
connect(m_outputDirInput, &LocationInputWithBrowseButton::clicked,
|
|
this, [this]() {
|
|
QString initPath = getOutputDir();
|
|
if (!QFileInfo::exists(initPath)) {
|
|
initPath = getDefaultOutputDir();
|
|
}
|
|
|
|
QString dirPath = QFileDialog::getExistingDirectory(this,
|
|
tr("Select Export Output Directory"),
|
|
initPath,
|
|
QFileDialog::ShowDirsOnly
|
|
| QFileDialog::DontResolveSymlinks);
|
|
|
|
if (!dirPath.isEmpty()) {
|
|
m_outputDirInput->setText(dirPath);
|
|
}
|
|
});
|
|
}
|
|
|
|
return box;
|
|
}
|
|
|
|
QGroupBox *ExportDialog::setupAdvancedGroup(QWidget *p_parent)
|
|
{
|
|
auto box = new QGroupBox(tr("Advanced"), p_parent);
|
|
auto layout = new QVBoxLayout(box);
|
|
|
|
m_advancedSettings.resize(AdvancedSettings::Max);
|
|
|
|
m_advancedSettings[AdvancedSettings::General] = setupGeneralAdvancedSettings(box);
|
|
layout->addWidget(m_advancedSettings[AdvancedSettings::General]);
|
|
|
|
return box;
|
|
}
|
|
|
|
QWidget *ExportDialog::setupGeneralAdvancedSettings(QWidget *p_parent)
|
|
{
|
|
QWidget *widget = new QWidget(p_parent);
|
|
auto layout = WidgetsFactory::createFormLayout(widget);
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
{
|
|
m_recursiveCheckBox = WidgetsFactory::createCheckBox(tr("Process sub-folders"), widget);
|
|
layout->addRow(m_recursiveCheckBox);
|
|
}
|
|
|
|
{
|
|
m_exportAttachmentsCheckBox = WidgetsFactory::createCheckBox(tr("Export attachments"), widget);
|
|
layout->addRow(m_exportAttachmentsCheckBox);
|
|
}
|
|
|
|
return widget;
|
|
}
|
|
|
|
void ExportDialog::setupButtonBox()
|
|
{
|
|
setDialogButtonBox(QDialogButtonBox::Close);
|
|
|
|
auto box = getDialogButtonBox();
|
|
|
|
m_exportBtn = box->addButton(tr("Export"), QDialogButtonBox::ActionRole);
|
|
connect(m_exportBtn, &QPushButton::clicked,
|
|
this, &ExportDialog::startExport);
|
|
|
|
m_openDirBtn = box->addButton(tr("Open Directory"), QDialogButtonBox::ActionRole);
|
|
connect(m_openDirBtn, &QPushButton::clicked,
|
|
this, [this]() {
|
|
auto dir = getOutputDir();
|
|
if (!dir.isEmpty()) {
|
|
WidgetUtils::openUrlByDesktop(QUrl::fromLocalFile(dir));
|
|
}
|
|
});
|
|
|
|
m_copyContentBtn = box->addButton(tr("Copy Content"), QDialogButtonBox::ActionRole);
|
|
m_copyContentBtn->setToolTip(tr("Copy exported file content"));
|
|
m_copyContentBtn->setEnabled(false);
|
|
connect(m_copyContentBtn, &QPushButton::clicked,
|
|
this, [this]() {
|
|
if (m_exportedFile.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
const auto content = FileUtils::readTextFile(m_exportedFile);
|
|
if (!content.isEmpty()) {
|
|
ClipboardUtils::setTextToClipboard(content);
|
|
}
|
|
});
|
|
}
|
|
|
|
QString ExportDialog::getOutputDir() const
|
|
{
|
|
return m_outputDirInput->text();
|
|
}
|
|
|
|
void ExportDialog::initOptions()
|
|
{
|
|
// Read it from config.
|
|
const auto &sessionConfig = ConfigMgr::getInst().getSessionConfig();
|
|
m_option = sessionConfig.getExportOption();
|
|
m_customOptions = sessionConfig.getCustomExportOptions();
|
|
|
|
const auto &theme = VNoteX::getInst().getThemeMgr().getCurrentTheme();
|
|
m_option.m_renderingStyleFile = theme.getFile(Theme::File::WebStyleSheet);
|
|
m_option.m_syntaxHighlightStyleFile = theme.getFile(Theme::File::HighlightStyleSheet);
|
|
|
|
if (m_option.m_outputDir.isEmpty()) {
|
|
m_option.m_outputDir = getDefaultOutputDir();
|
|
}
|
|
|
|
if (findCustomOption(m_option.m_customExport) == -1) {
|
|
m_option.m_customExport = m_customOptions.isEmpty() ? QString() : m_customOptions[0].m_name;
|
|
}
|
|
}
|
|
|
|
void ExportDialog::restoreFields(const ExportOption &p_option)
|
|
{
|
|
{
|
|
int idx = m_sourceComboBox->findData(static_cast<int>(p_option.m_source));
|
|
if (idx != -1) {
|
|
m_sourceComboBox->setCurrentIndex(idx);
|
|
}
|
|
}
|
|
|
|
{
|
|
int idx = m_targetFormatComboBox->findData(static_cast<int>(p_option.m_targetFormat));
|
|
if (idx != -1) {
|
|
m_targetFormatComboBox->setCurrentIndex(idx);
|
|
}
|
|
}
|
|
|
|
m_transparentBgCheckBox->setChecked(p_option.m_useTransparentBg);
|
|
|
|
{
|
|
int idx = m_renderingStyleComboBox->findData(p_option.m_renderingStyleFile);
|
|
if (idx != -1) {
|
|
m_renderingStyleComboBox->setCurrentIndex(idx);
|
|
}
|
|
}
|
|
|
|
{
|
|
int idx = m_syntaxHighlightStyleComboBox->findData(p_option.m_syntaxHighlightStyleFile);
|
|
if (idx != -1) {
|
|
m_syntaxHighlightStyleComboBox->setCurrentIndex(idx);
|
|
}
|
|
}
|
|
|
|
m_outputDirInput->setText(p_option.m_outputDir);
|
|
|
|
m_recursiveCheckBox->setChecked(p_option.m_recursive);
|
|
|
|
m_exportAttachmentsCheckBox->setChecked(p_option.m_exportAttachments);
|
|
}
|
|
|
|
void ExportDialog::saveFields(ExportOption &p_option)
|
|
{
|
|
p_option.m_source = static_cast<ExportSource>(m_sourceComboBox->currentData().toInt());
|
|
p_option.m_targetFormat = static_cast<ExportFormat>(m_targetFormatComboBox->currentData().toInt());
|
|
p_option.m_useTransparentBg = m_transparentBgCheckBox->isChecked();
|
|
p_option.m_renderingStyleFile = m_renderingStyleComboBox->currentData().toString();
|
|
p_option.m_syntaxHighlightStyleFile = m_syntaxHighlightStyleComboBox->currentData().toString();
|
|
p_option.m_outputDir = getOutputDir();
|
|
p_option.m_recursive = m_recursiveCheckBox->isChecked();
|
|
p_option.m_exportAttachments = m_exportAttachmentsCheckBox->isChecked();
|
|
|
|
if (m_advancedSettings[AdvancedSettings::HTML]) {
|
|
saveFields(p_option.m_htmlOption);
|
|
}
|
|
|
|
if (m_advancedSettings[AdvancedSettings::PDF]) {
|
|
saveFields(p_option.m_pdfOption);
|
|
}
|
|
|
|
if (m_advancedSettings[AdvancedSettings::Custom]) {
|
|
saveCustomFields(p_option);
|
|
}
|
|
}
|
|
|
|
void ExportDialog::startExport()
|
|
{
|
|
if (m_exportOngoing) {
|
|
return;
|
|
}
|
|
|
|
// On start.
|
|
{
|
|
clearInformationText();
|
|
m_exportedFile.clear();
|
|
m_exportOngoing = true;
|
|
updateUIOnExport();
|
|
}
|
|
|
|
saveFields(m_option);
|
|
|
|
int ret = doExport(m_option);
|
|
appendLog(tr("%n file(s) exported", "", ret));
|
|
|
|
// On end.
|
|
{
|
|
m_exportOngoing = false;
|
|
updateUIOnExport();
|
|
}
|
|
}
|
|
|
|
void ExportDialog::rejectedButtonClicked()
|
|
{
|
|
if (m_exportOngoing) {
|
|
// Just cancel the export.
|
|
appendLog(tr("Cancelling the export"));
|
|
getExporter()->stop();
|
|
} else {
|
|
Dialog::rejectedButtonClicked();
|
|
}
|
|
}
|
|
|
|
void ExportDialog::appendLog(const QString &p_log)
|
|
{
|
|
appendInformationText(">>> " + p_log);
|
|
QCoreApplication::sendPostedEvents();
|
|
}
|
|
|
|
void ExportDialog::updateUIOnExport()
|
|
{
|
|
m_exportBtn->setEnabled(!m_exportOngoing);
|
|
if (m_exportOngoing) {
|
|
m_progressBar->setMaximum(0);
|
|
m_progressBar->show();
|
|
} else {
|
|
m_progressBar->hide();
|
|
}
|
|
m_copyContentBtn->setEnabled(!m_exportedFile.isEmpty());
|
|
}
|
|
|
|
int ExportDialog::doExport(ExportOption p_option)
|
|
{
|
|
if (p_option.m_targetFormat == ExportFormat::PDF && p_option.m_pdfOption.m_useWkhtmltopdf) {
|
|
// Check wkhtmltopdf executable.
|
|
const auto &wkExePath = p_option.m_pdfOption.m_wkhtmltopdfExePath;
|
|
if (wkExePath.isEmpty() || !QFileInfo::exists(wkExePath)) {
|
|
appendLog(tr("Please specify a valid wkhtmltopdf executable file (%1)").arg(wkExePath));
|
|
return 0;
|
|
}
|
|
|
|
p_option.m_transformSvgToPngEnabled = true;
|
|
p_option.m_removeCodeToolBarEnabled = true;
|
|
} else if (p_option.m_targetFormat == ExportFormat::Custom) {
|
|
int optIdx = findCustomOption(p_option.m_customExport);
|
|
if (optIdx == -1) {
|
|
appendLog(tr("Please specify a valid scheme"));
|
|
return 0;
|
|
}
|
|
|
|
p_option.m_customOption = &m_customOptions[optIdx];
|
|
}
|
|
|
|
int exportedFilesCount = 0;
|
|
|
|
switch (p_option.m_source) {
|
|
case ExportSource::CurrentBuffer:
|
|
{
|
|
Q_ASSERT(m_buffer);
|
|
const auto outputFile = getExporter()->doExport(p_option, m_buffer);
|
|
exportedFilesCount = outputFile.isEmpty() ? 0 : 1;
|
|
if (exportedFilesCount == 1 && p_option.m_targetFormat == ExportFormat::HTML) {
|
|
m_exportedFile = outputFile;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ExportSource::CurrentNote:
|
|
{
|
|
Q_ASSERT(m_note);
|
|
const auto outputFile = getExporter()->doExport(p_option, m_note);
|
|
exportedFilesCount = outputFile.isEmpty() ? 0 : 1;
|
|
if (exportedFilesCount == 1 && p_option.m_targetFormat == ExportFormat::HTML) {
|
|
m_exportedFile = outputFile;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ExportSource::CurrentFolder:
|
|
{
|
|
Q_ASSERT(m_folder);
|
|
const auto outputFiles = getExporter()->doExportFolder(p_option, m_folder);
|
|
exportedFilesCount = outputFiles.size();
|
|
break;
|
|
}
|
|
|
|
case ExportSource::CurrentNotebook:
|
|
{
|
|
Q_ASSERT(m_notebook);
|
|
const auto outputFiles = getExporter()->doExport(p_option, m_notebook);
|
|
exportedFilesCount = outputFiles.size();
|
|
break;
|
|
}
|
|
}
|
|
|
|
return exportedFilesCount;
|
|
}
|
|
|
|
Exporter *ExportDialog::getExporter()
|
|
{
|
|
if (!m_exporter) {
|
|
m_exporter = new Exporter(this);
|
|
connect(m_exporter, &Exporter::progressUpdated,
|
|
this, &ExportDialog::updateProgress);
|
|
connect(m_exporter, &Exporter::logRequested,
|
|
this, &ExportDialog::appendLog);
|
|
}
|
|
return m_exporter;
|
|
}
|
|
|
|
void ExportDialog::updateProgress(int p_val, int p_maximum)
|
|
{
|
|
if (p_maximum < m_progressBar->value()) {
|
|
m_progressBar->setValue(p_val);
|
|
m_progressBar->setMaximum(p_maximum);
|
|
} else {
|
|
m_progressBar->setMaximum(p_maximum);
|
|
m_progressBar->setValue(p_val);
|
|
}
|
|
}
|
|
|
|
QWidget *ExportDialog::getHtmlAdvancedSettings()
|
|
{
|
|
if (!m_advancedSettings[AdvancedSettings::HTML]) {
|
|
// Setup HTML advanced settings.
|
|
QWidget *widget = new QWidget(m_advancedGroupBox);
|
|
auto layout = WidgetsFactory::createFormLayout(widget);
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
{
|
|
m_embedStylesCheckBox = WidgetsFactory::createCheckBox(tr("Embed styles"), widget);
|
|
layout->addRow(m_embedStylesCheckBox);
|
|
}
|
|
|
|
{
|
|
m_embedImagesCheckBox = WidgetsFactory::createCheckBox(tr("Embed images"), widget);
|
|
layout->addRow(m_embedImagesCheckBox);
|
|
}
|
|
|
|
{
|
|
m_completePageCheckBox = WidgetsFactory::createCheckBox(tr("Complete page"), widget);
|
|
m_completePageCheckBox->setToolTip(tr("Export the whole page along with images which may change the links structure"));
|
|
connect(m_completePageCheckBox, &QCheckBox::stateChanged,
|
|
this, [this](int p_state) {
|
|
bool checked = p_state == Qt::Checked;
|
|
m_embedImagesCheckBox->setEnabled(checked);
|
|
});
|
|
layout->addRow(m_completePageCheckBox);
|
|
}
|
|
|
|
{
|
|
m_useMimeHtmlFormatCheckBox = WidgetsFactory::createCheckBox(tr("Mime HTML format"), widget);
|
|
connect(m_useMimeHtmlFormatCheckBox, &QCheckBox::stateChanged,
|
|
this, [this](int p_state) {
|
|
bool checked = p_state == Qt::Checked;
|
|
m_embedStylesCheckBox->setEnabled(!checked);
|
|
m_completePageCheckBox->setEnabled(!checked);
|
|
});
|
|
// TODO: do not support MHTML for now.
|
|
m_useMimeHtmlFormatCheckBox->setEnabled(false);
|
|
m_useMimeHtmlFormatCheckBox->hide();
|
|
layout->addRow(m_useMimeHtmlFormatCheckBox);
|
|
}
|
|
|
|
{
|
|
m_addOutlinePanelCheckBox = WidgetsFactory::createCheckBox(tr("Add outline panel"), widget);
|
|
layout->addRow(m_addOutlinePanelCheckBox);
|
|
}
|
|
|
|
m_advancedGroupBox->layout()->addWidget(widget);
|
|
|
|
m_advancedSettings[AdvancedSettings::HTML] = widget;
|
|
|
|
restoreFields(m_option.m_htmlOption);
|
|
}
|
|
|
|
return m_advancedSettings[AdvancedSettings::HTML];
|
|
}
|
|
|
|
void ExportDialog::showAdvancedSettings(AdvancedSettings p_settings)
|
|
{
|
|
for (int i = AdvancedSettings::General + 1; i < m_advancedSettings.size(); ++i) {
|
|
if (m_advancedSettings[i]) {
|
|
m_advancedSettings[i]->hide();
|
|
}
|
|
}
|
|
|
|
QWidget *widget = nullptr;
|
|
switch (p_settings) {
|
|
case AdvancedSettings::HTML:
|
|
widget = getHtmlAdvancedSettings();
|
|
break;
|
|
|
|
case AdvancedSettings::PDF:
|
|
widget = getPdfAdvancedSettings();
|
|
break;
|
|
|
|
case AdvancedSettings::Custom:
|
|
widget = getCustomAdvancedSettings();
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (widget) {
|
|
widget->show();
|
|
}
|
|
}
|
|
|
|
void ExportDialog::restoreFields(const ExportHtmlOption &p_option)
|
|
{
|
|
m_embedStylesCheckBox->setChecked(p_option.m_embedStyles);
|
|
m_embedImagesCheckBox->setChecked(p_option.m_embedImages);
|
|
m_completePageCheckBox->setChecked(p_option.m_completePage);
|
|
m_useMimeHtmlFormatCheckBox->setChecked(p_option.m_useMimeHtmlFormat);
|
|
m_addOutlinePanelCheckBox->setChecked(p_option.m_addOutlinePanel);
|
|
}
|
|
|
|
void ExportDialog::saveFields(ExportHtmlOption &p_option)
|
|
{
|
|
p_option.m_embedStyles = m_embedStylesCheckBox->isChecked();
|
|
p_option.m_embedImages = m_embedImagesCheckBox->isChecked();
|
|
p_option.m_completePage = m_completePageCheckBox->isChecked();
|
|
p_option.m_useMimeHtmlFormat = m_useMimeHtmlFormatCheckBox->isChecked();
|
|
p_option.m_addOutlinePanel = m_addOutlinePanelCheckBox->isChecked();
|
|
}
|
|
|
|
QWidget *ExportDialog::getPdfAdvancedSettings()
|
|
{
|
|
if (!m_advancedSettings[AdvancedSettings::PDF]) {
|
|
QWidget *widget = new QWidget(m_advancedGroupBox);
|
|
auto layout = WidgetsFactory::createFormLayout(widget);
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
{
|
|
m_pageLayoutBtn = new QPushButton(tr("Settings"), widget);
|
|
connect(m_pageLayoutBtn, &QPushButton::clicked,
|
|
this, [this]() {
|
|
QPrinter printer;
|
|
printer.setPageLayout(*m_pageLayout);
|
|
|
|
QPageSetupDialog dlg(&printer, this);
|
|
if (dlg.exec() != QDialog::Accepted) {
|
|
return;
|
|
}
|
|
|
|
m_pageLayout->setUnits(QPageLayout::Millimeter);
|
|
m_pageLayout->setPageSize(printer.pageLayout().pageSize());
|
|
m_pageLayout->setMargins(printer.pageLayout().margins(QPageLayout::Millimeter));
|
|
m_pageLayout->setOrientation(printer.pageLayout().orientation());
|
|
|
|
updatePageLayoutButtonLabel();
|
|
});
|
|
layout->addRow(tr("Page layout:"), m_pageLayoutBtn);
|
|
}
|
|
|
|
{
|
|
m_addTableOfContentsCheckBox = WidgetsFactory::createCheckBox(tr("Add Table-of-Contents"), widget);
|
|
layout->addRow(m_addTableOfContentsCheckBox);
|
|
}
|
|
|
|
{
|
|
auto useLayout = new QHBoxLayout();
|
|
|
|
m_useWkhtmltopdfCheckBox = WidgetsFactory::createCheckBox(tr("Use wkhtmltopdf (outline supported)"), widget);
|
|
useLayout->addWidget(m_useWkhtmltopdfCheckBox);
|
|
|
|
auto downloadBtn = new QPushButton(tr("Download"), widget);
|
|
connect(downloadBtn, &QPushButton::clicked,
|
|
this, []() {
|
|
WidgetUtils::openUrlByDesktop(QUrl("https://wkhtmltopdf.org/downloads.html"));
|
|
});
|
|
useLayout->addWidget(downloadBtn);
|
|
|
|
layout->addRow(useLayout);
|
|
}
|
|
|
|
{
|
|
m_allInOneCheckBox = WidgetsFactory::createCheckBox(tr("All-in-One"), widget);
|
|
m_allInOneCheckBox->setToolTip(tr("Export all source files into one file"));
|
|
m_allInOneCheckBox->setEnabled(false);
|
|
connect(m_useWkhtmltopdfCheckBox, &QCheckBox::stateChanged,
|
|
this, [this](int p_state) {
|
|
m_allInOneCheckBox->setEnabled(p_state == Qt::Checked);
|
|
});
|
|
layout->addRow(m_allInOneCheckBox);
|
|
}
|
|
|
|
{
|
|
auto pathLayout = new QHBoxLayout();
|
|
|
|
m_wkhtmltopdfExePathLineEdit = WidgetsFactory::createLineEdit(widget);
|
|
pathLayout->addWidget(m_wkhtmltopdfExePathLineEdit);
|
|
|
|
auto browseBtn = new QPushButton(tr("Browse"), widget);
|
|
pathLayout->addWidget(browseBtn);
|
|
connect(browseBtn, &QPushButton::clicked,
|
|
this, [this]() {
|
|
QString filePath = QFileDialog::getOpenFileName(this,
|
|
tr("Select wkhtmltopdf Executable"),
|
|
QCoreApplication::applicationDirPath());
|
|
|
|
if (!filePath.isEmpty()) {
|
|
m_wkhtmltopdfExePathLineEdit->setText(filePath);
|
|
}
|
|
});
|
|
|
|
layout->addRow(tr("Wkhtmltopdf path:"), pathLayout);
|
|
}
|
|
|
|
{
|
|
m_wkhtmltopdfArgsLineEdit = WidgetsFactory::createLineEdit(widget);
|
|
layout->addRow(tr("Wkhtmltopdf arguments:"), m_wkhtmltopdfArgsLineEdit);
|
|
}
|
|
|
|
m_advancedGroupBox->layout()->addWidget(widget);
|
|
|
|
m_advancedSettings[AdvancedSettings::PDF] = widget;
|
|
|
|
restoreFields(m_option.m_pdfOption);
|
|
}
|
|
|
|
return m_advancedSettings[AdvancedSettings::PDF];
|
|
}
|
|
|
|
QWidget *ExportDialog::getCustomAdvancedSettings()
|
|
{
|
|
if (!m_advancedSettings[AdvancedSettings::Custom]) {
|
|
QWidget *widget = new QWidget(m_advancedGroupBox);
|
|
auto layout = WidgetsFactory::createFormLayout(widget);
|
|
layout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
{
|
|
auto schemeLayout = new QHBoxLayout();
|
|
layout->addRow(tr("Scheme:"), schemeLayout);
|
|
|
|
m_customExportComboBox = WidgetsFactory::createComboBox(widget);
|
|
m_customExportComboBox->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
|
|
schemeLayout->addWidget(m_customExportComboBox, 1);
|
|
|
|
auto addBtn = new QPushButton(tr("New"), widget);
|
|
addBtn->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
|
|
connect(addBtn, &QPushButton::clicked,
|
|
this, &ExportDialog::addCustomExportScheme);
|
|
schemeLayout->addWidget(addBtn);
|
|
|
|
auto delBtn = new QPushButton(tr("Delete"), widget);
|
|
delBtn->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
|
|
connect(delBtn, &QPushButton::clicked,
|
|
this, &ExportDialog::removeCustomExportScheme);
|
|
schemeLayout->addWidget(delBtn);
|
|
}
|
|
|
|
{
|
|
m_targetSuffixLineEdit = WidgetsFactory::createLineEdit(widget);
|
|
m_targetSuffixLineEdit->setToolTip(tr("Suffix of the target file like docs/pdf/epub"));
|
|
m_targetSuffixLineEdit->setEnabled(false);
|
|
layout->addRow(tr("Target file suffix:"), m_targetSuffixLineEdit);
|
|
}
|
|
|
|
{
|
|
m_resourcePathSeparatorLineEdit = WidgetsFactory::createLineEdit(widget);
|
|
m_resourcePathSeparatorLineEdit->setToolTip(tr("Separator used to concatenate resource folder paths"));
|
|
m_resourcePathSeparatorLineEdit->setEnabled(false);
|
|
layout->addRow(tr("Resource path separator:"), m_resourcePathSeparatorLineEdit);
|
|
}
|
|
|
|
{
|
|
m_useHtmlInputCheckBox = WidgetsFactory::createCheckBox(tr("Use HTML format as input"), widget);
|
|
m_useHtmlInputCheckBox->setToolTip(tr("Convert to HTMl format first as the input of the custom export command"));
|
|
m_useHtmlInputCheckBox->setEnabled(false);
|
|
layout->addRow(m_useHtmlInputCheckBox);
|
|
}
|
|
|
|
{
|
|
m_allInOneCheckBox = WidgetsFactory::createCheckBox(tr("All-in-One"), widget);
|
|
m_allInOneCheckBox->setToolTip(tr("Export all source files into one file"));
|
|
m_allInOneCheckBox->setEnabled(false);
|
|
layout->addRow(m_allInOneCheckBox);
|
|
}
|
|
|
|
{
|
|
m_targetPageScrollableCheckBox = WidgetsFactory::createCheckBox(tr("Target page scrollable"), widget);
|
|
m_targetPageScrollableCheckBox->setToolTip(tr("Whether the page of the target file is scrollable"));
|
|
m_targetPageScrollableCheckBox->setEnabled(false);
|
|
layout->addRow(m_targetPageScrollableCheckBox);
|
|
}
|
|
|
|
{
|
|
auto usage = tr("Command:\n"
|
|
"\t%1: List of input files.\n"
|
|
"\t%2: List of paths to search for images and other resources.\n"
|
|
"\t%3: Path of rendering CSS style sheet.\n"
|
|
"\t%4: Path of syntax highlighting CSS style sheet.\n"
|
|
"\t%5: Path of output file.");
|
|
layout->addRow(new QLabel(usage, widget));
|
|
}
|
|
|
|
{
|
|
m_commandTextEdit = WidgetsFactory::createPlainTextEdit(widget);
|
|
#if defined(Q_OS_WIN)
|
|
m_commandTextEdit->setPlaceholderText("pandoc.exe --resource-path=.;%2 --css=%3 --css=%4 -s -o %5 %1");
|
|
#else
|
|
m_commandTextEdit->setPlaceholderText("pandoc --resource-path=.:%2 --css=%3 --css=%4 -s -o %5 %1");
|
|
#endif
|
|
m_commandTextEdit->setMaximumHeight(m_commandTextEdit->minimumSizeHint().height());
|
|
m_commandTextEdit->setEnabled(false);
|
|
layout->addRow(m_commandTextEdit);
|
|
}
|
|
|
|
connect(m_customExportComboBox, QOverload<int>::of(&QComboBox::currentIndexChanged),
|
|
this, &ExportDialog::customExportCurrentSchemeChanged);
|
|
|
|
m_advancedGroupBox->layout()->addWidget(widget);
|
|
|
|
m_advancedSettings[AdvancedSettings::Custom] = widget;
|
|
|
|
restoreCustomFields(m_option);
|
|
}
|
|
|
|
return m_advancedSettings[AdvancedSettings::Custom];
|
|
}
|
|
|
|
void ExportDialog::restoreFields(const ExportPdfOption &p_option)
|
|
{
|
|
m_pageLayout = p_option.m_layout;
|
|
updatePageLayoutButtonLabel();
|
|
|
|
m_addTableOfContentsCheckBox->setChecked(p_option.m_addTableOfContents);
|
|
m_useWkhtmltopdfCheckBox->setChecked(p_option.m_useWkhtmltopdf);
|
|
m_allInOneCheckBox->setChecked(p_option.m_allInOne);
|
|
m_wkhtmltopdfExePathLineEdit->setText(p_option.m_wkhtmltopdfExePath);
|
|
m_wkhtmltopdfArgsLineEdit->setText(p_option.m_wkhtmltopdfArgs);
|
|
}
|
|
|
|
void ExportDialog::saveFields(ExportPdfOption &p_option)
|
|
{
|
|
p_option.m_layout = m_pageLayout;
|
|
p_option.m_addTableOfContents = m_addTableOfContentsCheckBox->isChecked();
|
|
p_option.m_useWkhtmltopdf = m_useWkhtmltopdfCheckBox->isChecked();
|
|
p_option.m_allInOne = m_allInOneCheckBox->isChecked();
|
|
p_option.m_wkhtmltopdfExePath = m_wkhtmltopdfExePathLineEdit->text();
|
|
p_option.m_wkhtmltopdfArgs = m_wkhtmltopdfArgsLineEdit->text();
|
|
}
|
|
|
|
void ExportDialog::restoreCustomFields(const ExportOption &p_option)
|
|
{
|
|
m_customExportComboBox->clear();
|
|
int curIndex = -1;
|
|
for (int i = 0; i < m_customOptions.size(); ++i) {
|
|
m_customExportComboBox->addItem(m_customOptions[i].m_name, m_customOptions[i].m_name);
|
|
if (m_customOptions[i].m_name == p_option.m_customExport) {
|
|
curIndex = i;
|
|
}
|
|
}
|
|
m_customExportComboBox->setCurrentIndex(curIndex);
|
|
}
|
|
|
|
void ExportDialog::saveCustomFields(ExportOption &p_option)
|
|
{
|
|
p_option.m_customExport = m_customExportComboBox->currentData().toString();
|
|
|
|
int idx = findCustomOption(p_option.m_customExport);
|
|
if (idx > -1) {
|
|
auto &opt = m_customOptions[idx];
|
|
opt.m_targetSuffix = m_targetSuffixLineEdit->text();
|
|
opt.m_resourcePathSeparator = m_resourcePathSeparatorLineEdit->text();
|
|
opt.m_useHtmlInput = m_useHtmlInputCheckBox->isChecked();
|
|
opt.m_allInOne = m_allInOneCheckBox->isChecked();
|
|
opt.m_targetPageScrollable = m_targetPageScrollableCheckBox->isChecked();
|
|
|
|
opt.m_command = m_commandTextEdit->toPlainText().trimmed();
|
|
int lineIdx = opt.m_command.indexOf(QLatin1Char('\n'));
|
|
if (lineIdx > -1) {
|
|
opt.m_command = opt.m_command.left(lineIdx);
|
|
}
|
|
|
|
ConfigMgr::getInst().getSessionConfig().setCustomExportOptions(m_customOptions);
|
|
}
|
|
}
|
|
|
|
void ExportDialog::updatePageLayoutButtonLabel()
|
|
{
|
|
Q_ASSERT(m_pageLayout);
|
|
m_pageLayoutBtn->setText(
|
|
QStringLiteral("%1, %2").arg(m_pageLayout->pageSize().name(),
|
|
m_pageLayout->orientation() == QPageLayout::Portrait ? tr("Portrait") : tr("Landscape")));
|
|
}
|
|
|
|
int ExportDialog::findCustomOption(const QString &p_name) const
|
|
{
|
|
if (p_name.isEmpty()) {
|
|
return -1;
|
|
}
|
|
|
|
for (int i = 0; i < m_customOptions.size(); ++i) {
|
|
if (m_customOptions[i].m_name == p_name) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
void ExportDialog::addCustomExportScheme()
|
|
{
|
|
QString name;
|
|
while (true) {
|
|
name = QInputDialog::getText(this, tr("New Custom Export Scheme"), tr("Scheme name:"));
|
|
if (name.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
if (findCustomOption(name) != -1) {
|
|
MessageBoxHelper::notify(MessageBoxHelper::Warning,
|
|
tr("Name conflicts with existing scheme."),
|
|
this);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Based on current scheme.
|
|
ExportCustomOption newOption;
|
|
|
|
{
|
|
int curIndex = findCustomOption(m_customExportComboBox->currentData().toString());
|
|
if (curIndex > -1) {
|
|
newOption = m_customOptions[curIndex];
|
|
}
|
|
}
|
|
|
|
newOption.m_name = name;
|
|
m_customOptions.append(newOption);
|
|
ConfigMgr::getInst().getSessionConfig().setCustomExportOptions(m_customOptions);
|
|
|
|
// Add it to combo box.
|
|
m_customExportComboBox->addItem(name, name);
|
|
m_customExportComboBox->setCurrentIndex(m_customExportComboBox->findData(name));
|
|
}
|
|
|
|
void ExportDialog::removeCustomExportScheme()
|
|
{
|
|
auto name = m_customExportComboBox->currentData().toString();
|
|
if (name.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
int ret = MessageBoxHelper::questionOkCancel(MessageBoxHelper::Warning,
|
|
tr("Delete scheme (%1)?").arg(name),
|
|
QString(),
|
|
QString(),
|
|
this);
|
|
if (ret != QMessageBox::Ok) {
|
|
return;
|
|
}
|
|
|
|
int idx = findCustomOption(name);
|
|
Q_ASSERT(idx > -1);
|
|
m_customOptions.remove(idx);
|
|
ConfigMgr::getInst().getSessionConfig().setCustomExportOptions(m_customOptions);
|
|
|
|
m_customExportComboBox->removeItem(m_customExportComboBox->currentIndex());
|
|
}
|
|
|
|
void ExportDialog::customExportCurrentSchemeChanged(int p_comboIdx)
|
|
{
|
|
const bool enabled = p_comboIdx >= 0;
|
|
m_targetSuffixLineEdit->setEnabled(enabled);
|
|
m_resourcePathSeparatorLineEdit->setEnabled(enabled);
|
|
m_useHtmlInputCheckBox->setEnabled(enabled);
|
|
m_allInOneCheckBox->setEnabled(enabled);
|
|
m_targetPageScrollableCheckBox->setEnabled(enabled);
|
|
m_commandTextEdit->setEnabled(enabled);
|
|
|
|
if (p_comboIdx < 0) {
|
|
m_option.m_customExport.clear();
|
|
return;
|
|
}
|
|
|
|
auto name = m_customExportComboBox->currentData().toString();
|
|
m_option.m_customExport = name;
|
|
int curIndex = findCustomOption(name);
|
|
Q_ASSERT(curIndex > -1);
|
|
const auto &opt = m_customOptions[curIndex];
|
|
m_targetSuffixLineEdit->setText(opt.m_targetSuffix);
|
|
m_resourcePathSeparatorLineEdit->setText(opt.m_resourcePathSeparator);
|
|
m_useHtmlInputCheckBox->setChecked(opt.m_useHtmlInput);
|
|
m_allInOneCheckBox->setChecked(opt.m_allInOne);
|
|
m_targetPageScrollableCheckBox->setChecked(opt.m_targetPageScrollable);
|
|
m_commandTextEdit->setPlainText(opt.m_command);
|
|
}
|