/*
 * Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com)
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <https://www.gnu.org/licenses/>.
 */
#include "toolsgeneralwidget.h"
#include "ui_toolsgeneralwidget.h"
#include "../mainwindow.h"
#include "../settings.h"
#include "../iconsmanager.h"

#include <QFileDialog>
#include <QMessageBox>

ToolsGeneralWidget::ToolsGeneralWidget(const QString &name, const QString &group, QWidget *parent) :
    SettingsWidget(name,group,parent),
    ui(new Ui::ToolsGeneralWidget)
{
    ui->setupUi(this);
    ui->cbMacros->setModel(&mMacroInfoModel);
    QItemSelectionModel *m=ui->lstTools->selectionModel();
    ui->lstTools->setModel(&mToolsModel);
    delete m;
    mEditType = EditType::None;
    showEditPanel(false);
    connect(ui->lstTools->selectionModel(), &QItemSelectionModel::currentRowChanged,
            this,&ToolsGeneralWidget::onToolsCurrentChanged);
    connect(ui->txtProgram,&QLineEdit::textChanged,
            this, &ToolsGeneralWidget::updateDemo);
    connect(ui->txtParameters,&QLineEdit::textChanged,
            this, &ToolsGeneralWidget::updateDemo);

    connect(ui->txtTitle,&QLineEdit::textChanged,
            this, &ToolsGeneralWidget::onEdited);
    connect(ui->txtProgram,&QLineEdit::textChanged,
            this, &ToolsGeneralWidget::onEdited);
    connect(ui->txtParameters,&QLineEdit::textChanged,
            this, &ToolsGeneralWidget::onEdited);
    connect(ui->txtDirectory,&QLineEdit::textChanged,
            this, &ToolsGeneralWidget::onEdited);
    connect(ui->chkPauseConsole,&QCheckBox::stateChanged,
            this, &ToolsGeneralWidget::onEdited);
}

ToolsGeneralWidget::~ToolsGeneralWidget()
{
    delete ui;
}

void ToolsGeneralWidget::onToolsCurrentChanged(const QModelIndex &current, const QModelIndex &previous)
{
    finishEditing(true,previous);
    QModelIndex index = current;
    if (!index.isValid())
        return;
    PToolItem item = mToolsModel.getTool(index.row());
    if (item) {
        prepareEdit(item);
    }
}

void ToolsGeneralWidget::finishEditing(bool askSave, const QModelIndex& itemIndex)
{
    auto action = finally([this]{
        showEditPanel(false);
    });
    if (mEditType == EditType::None)
        return;
    if (!mEdited)
        return;
    if (askSave && QMessageBox::question(this,
                          tr("Save Changes?"),
                          tr("Do you want to save changes to \"%1\"?").arg(ui->txtTitle->text()),
                          QMessageBox::Yes | QMessageBox::No,
                          QMessageBox::Yes) != QMessageBox::Yes) {
        return;
    }
    if (ui->txtTitle->text().isEmpty()) {
        QMessageBox::critical(this,
                              tr("Error"),
                              tr("Title shouldn't be empty!"));
        return;
    }
    mEditType = EditType::None;
    QModelIndex index=itemIndex.isValid()?itemIndex:ui->lstTools->currentIndex();
    if (!index.isValid())
        return;

    PToolItem item = mToolsModel.getTool(index.row());
    item->workingDirectory = ui->txtDirectory->text();
    item->parameters = ui->txtParameters->text();
    item->program = ui->txtProgram->text();
    item->title = ui->txtTitle->text();
    item->pauseAfterExit = ui->chkPauseConsole->isChecked();
    mEdited=false;
}

void ToolsGeneralWidget::prepareEdit(const PToolItem& item)
{
    mEditType = EditType::Edit;
    ui->txtDirectory->setText(item->workingDirectory);
    ui->txtParameters->setText(item->parameters);
    ui->txtProgram->setText(item->program);
    ui->txtTitle->setText(item->title);
    ui->chkPauseConsole->setChecked(item->pauseAfterExit);
    showEditPanel(true);
    ui->txtTitle->setFocus();
    mEdited = false;
}

void ToolsGeneralWidget::showEditPanel(bool isShow)
{
    ui->panelEdit->setVisible(isShow);
    ui->panelEditButtons->setVisible(!isShow);
}

void ToolsGeneralWidget::onEdited()
{
    mEdited=true;
}

void ToolsGeneralWidget::updateDemo()
{
    ui->txtDemo->setText(
                parseMacros(ui->txtProgram->text())+ " " +
                parseMacros(ui->txtParameters->text()));
}

ToolsModel::ToolsModel(QObject *parent):QAbstractListModel(parent)
{

}

const QList<PToolItem> &ToolsModel::tools() const
{
    return mTools;
}

void ToolsModel::setTools(const QList<PToolItem> &newTools)
{
    beginResetModel();
    mTools = newTools;
    endResetModel();
}

void ToolsModel::addTool(PToolItem item)
{
    beginInsertRows(QModelIndex(),mTools.count(),mTools.count());
    mTools.append(item);
    endInsertRows();
}

PToolItem ToolsModel::getTool(int index)
{
    return mTools[index];
}

void ToolsModel::removeTool(int index)
{
    mTools.removeAt(index);
}

int ToolsModel::rowCount(const QModelIndex &) const
{
    return mTools.count();
}

QVariant ToolsModel::data(const QModelIndex &index, int role) const
{
    if (!index.isValid())
        return QVariant();
    if (role==Qt::DisplayRole) {
        PToolItem item = mTools[index.row()];
        return item->title;
    }
    return QVariant();
}

void ToolsGeneralWidget::on_btnAdd_clicked()
{
    ui->lstTools->setCurrentIndex(QModelIndex());
    PToolItem item = std::make_shared<ToolItem>();
    item->title = tr("untitled");
    item->pauseAfterExit = false;
    mToolsModel.addTool(item);
    QModelIndex index=mToolsModel.index(mToolsModel.tools().count()-1);
    ui->lstTools->setCurrentIndex(index);
    prepareEdit(item);
}


void ToolsGeneralWidget::on_btnEditOk_clicked()
{
    finishEditing(false);
}


void ToolsGeneralWidget::on_btnEditCancel_clicked()
{
    mEditType = EditType::None;
    showEditPanel(false);
}

void ToolsGeneralWidget::doLoad()
{
    mToolsModel.setTools(pMainWindow->toolsManager()->tools());
}

void ToolsGeneralWidget::doSave()
{
    finishEditing(true);
    pMainWindow->toolsManager()->setTools(mToolsModel.tools());
    pMainWindow->toolsManager()->save();
    pMainWindow->updateTools();
}

void ToolsGeneralWidget::updateIcons(const QSize &)
{
    pIconsManager->setIcon(ui->btnAdd,IconsManager::ACTION_MISC_ADD);
    pIconsManager->setIcon(ui->btnRemove,IconsManager::ACTION_MISC_REMOVE);
    pIconsManager->setIcon(ui->btnBrowseProgram,IconsManager::ACTION_FILE_OPEN_FOLDER);
    pIconsManager->setIcon(ui->btnBrowseWorkingDirectory,IconsManager::ACTION_FILE_OPEN_FOLDER);
}


void ToolsGeneralWidget::on_btnRemove_clicked()
{
    mEditType = EditType::None;
    finishEditing(false);
    QModelIndex index = ui->lstTools->currentIndex();
    if (index.isValid()) {
        mToolsModel.removeTool(index.row());
    }
}


void ToolsGeneralWidget::on_btnInsertMacro_clicked()
{
    ui->txtParameters->setText(
                ui->txtParameters->text() +
                ui->cbMacros->currentData(Qt::UserRole).toString());
}

void ToolsGeneralWidget::on_btnBrowseWorkingDirectory_clicked()
{
    QString folder = QFileDialog::getExistingDirectory(this,tr("Choose Folder"));
    if (!folder.isEmpty()) {
        ui->txtDirectory->setText(folder);
    }
}


void ToolsGeneralWidget::on_btnBrowseProgram_clicked()
{
    QString fileName = QFileDialog::getOpenFileName(
                this,
                tr("Select program"),
                pSettings->dirs().appDir(),
                tr("Executable files (*.exe)"));
    if (!fileName.isEmpty() ) {
        ui->txtProgram->setText(fileName);
    }
}