493 Zeilen
16 KiB
C++
493 Zeilen
16 KiB
C++
/**
|
|
* client/src/widget/objectlist/objectlistwidget.cpp
|
|
*
|
|
* This file is part of the traintastic source code.
|
|
*
|
|
* Copyright (C) 2019-2025 Reinder Feenstra
|
|
*
|
|
* 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 2
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "objectlistwidget.hpp"
|
|
#include <QVBoxLayout>
|
|
#include <QToolBar>
|
|
#include <QTableView>
|
|
#include <traintastic/locale/locale.hpp>
|
|
#include <traintastic/enum/outputchannel.hpp>
|
|
#include "../tablewidget.hpp"
|
|
#include "../../network/connection.hpp"
|
|
#include "../../network/object.hpp"
|
|
#include "../../network/method.hpp"
|
|
#include "../../network/callmethod.hpp"
|
|
#include "../../theme/theme.hpp"
|
|
#include "../../misc/methodaction.hpp"
|
|
#include "../../dialog/objectselectlistdialog.hpp"
|
|
#include "../../utils/enum.hpp"
|
|
|
|
|
|
#include "../../mainwindow.hpp"
|
|
|
|
|
|
|
|
#include <QMenu>
|
|
#include <QToolButton>
|
|
|
|
|
|
|
|
ObjectListWidget::ObjectListWidget(const ObjectPtr& object_, QWidget* parent) :
|
|
ListWidget(object_, parent),
|
|
m_requestIdInputMonitor{Connection::invalidRequestId},
|
|
m_requestIdOutputKeyboard{Connection::invalidRequestId},
|
|
m_toolbar{new QToolBar()}
|
|
{
|
|
if(Method* method = object()->getMethod("create");
|
|
method && method->resultType() == ValueType::Object)
|
|
{
|
|
if(method->argumentTypes().size() == 0) // Create method witout argument
|
|
{
|
|
m_actionCreate = m_toolbar->addAction(Theme::getIcon("add"), method->displayName(),
|
|
[this, method]()
|
|
{
|
|
if(m_requestIdCreate != Connection::invalidRequestId)
|
|
object()->connection()->cancelRequest(m_requestIdCreate);
|
|
|
|
m_requestIdCreate = method->call(
|
|
[this](const ObjectPtr& addedObject, std::optional<const Error> /*error*/)
|
|
{
|
|
m_requestIdCreate = Connection::invalidRequestId;
|
|
if(addedObject)
|
|
{
|
|
objectCreated(addedObject);
|
|
}
|
|
// TODO: show error
|
|
});
|
|
});
|
|
m_actionCreate->setEnabled(method->getAttributeBool(AttributeName::Enabled, true));
|
|
connect(method, &Method::attributeChanged, this,
|
|
[this](AttributeName name, QVariant value)
|
|
{
|
|
if(name == AttributeName::Enabled)
|
|
m_actionCreate->setEnabled(value.toBool());
|
|
});
|
|
}
|
|
else if(method->argumentTypes().size() == 1 && method->argumentTypes()[0] == ValueType::String)
|
|
{
|
|
m_buttonCreate = new QToolButton(m_toolbar);
|
|
m_buttonCreate->setIcon(Theme::getIcon("add"));
|
|
m_buttonCreate->setText(method->displayName());
|
|
m_buttonCreate->setPopupMode(QToolButton::InstantPopup);
|
|
|
|
QMenu* menu = new QMenu(m_buttonCreate);
|
|
|
|
QStringList classList = method->getAttribute(AttributeName::ClassList, QVariant()).toStringList();
|
|
for(const QString& classId : classList)
|
|
{
|
|
QAction* action = menu->addAction(Locale::tr("class_id:" + classId));
|
|
action->setData(classId);
|
|
connect(action, &QAction::triggered, this,
|
|
[this, method, action]()
|
|
{
|
|
if(m_requestIdCreate != Connection::invalidRequestId)
|
|
object()->connection()->cancelRequest(m_requestIdCreate);
|
|
|
|
m_requestIdCreate = method->call(action->data().toString(),
|
|
[this](const ObjectPtr& addedObject, std::optional<const Error> /*error*/)
|
|
{
|
|
m_requestIdCreate = Connection::invalidRequestId;
|
|
if(addedObject)
|
|
{
|
|
objectCreated(addedObject);
|
|
}
|
|
// TODO: show error
|
|
});
|
|
});
|
|
}
|
|
|
|
if(object_->classId() == "list.interface")
|
|
{
|
|
menu->addSeparator();
|
|
menu->addAction(Theme::getIcon("wizard"), Locale::tr("list:setup_using_wizard") + "...",
|
|
[]()
|
|
{
|
|
MainWindow::instance->showAddInterfaceWizard();
|
|
});
|
|
}
|
|
|
|
m_buttonCreate->setMenu(menu);
|
|
|
|
m_toolbar->addWidget(m_buttonCreate);
|
|
|
|
m_buttonCreate->setEnabled(method->getAttributeBool(AttributeName::Enabled, true));
|
|
connect(method, &Method::attributeChanged, this,
|
|
[this](AttributeName name, QVariant value)
|
|
{
|
|
if(name == AttributeName::Enabled)
|
|
m_buttonCreate->setEnabled(value.toBool());
|
|
});
|
|
}
|
|
else
|
|
Q_ASSERT(false); // unsupported method prototype
|
|
}
|
|
|
|
if(Method* method = object()->getMethod("add");
|
|
method &&
|
|
method->argumentTypes().size() == 1 && method->argumentTypes()[0] == ValueType::Object &&
|
|
method->resultType() == ValueType::Invalid)
|
|
{
|
|
const bool multiSelect = object_->classId() == "list.train_vehicle";
|
|
m_actionAdd = m_toolbar->addAction(Theme::getIcon("add"), method->displayName(),
|
|
[this, method, multiSelect]()
|
|
{
|
|
std::make_unique<ObjectSelectListDialog>(*method, multiSelect, this)->exec();
|
|
});
|
|
m_actionAdd->setEnabled(method->getAttributeBool(AttributeName::Enabled, true));
|
|
connect(method, &Method::attributeChanged, this,
|
|
[this](AttributeName name, QVariant value)
|
|
{
|
|
if(name == AttributeName::Enabled)
|
|
m_actionAdd->setEnabled(value.toBool());
|
|
});
|
|
}
|
|
|
|
if(hasEdit())
|
|
{
|
|
m_actionEdit = m_toolbar->addAction(Theme::getIcon("edit"), Locale::tr("list:edit"),
|
|
[this]()
|
|
{
|
|
for(const QString& id : getSelectedObjectIds())
|
|
MainWindow::instance->showObject(id);
|
|
});
|
|
m_actionEdit->setEnabled(false);
|
|
}
|
|
|
|
if(Method* method = object()->getMethod("remove"))
|
|
{
|
|
m_actionRemove = new MethodAction(Theme::getIcon("remove"), *method,
|
|
[this]()
|
|
{
|
|
for(const QString& id : getSelectedObjectIds())
|
|
callMethod(m_actionRemove->method(), nullptr, id);
|
|
});
|
|
m_actionRemove->setForceDisabled(true);
|
|
m_toolbar->addAction(m_actionRemove);
|
|
}
|
|
|
|
if(Method* method = object()->getMethod("delete"))
|
|
{
|
|
m_actionDelete = new MethodAction(Theme::getIcon("delete"), *method,
|
|
[this]()
|
|
{
|
|
for(const QString& id : getSelectedObjectIds())
|
|
callMethod(m_actionDelete->method(),
|
|
[this](std::optional<const Error> error)
|
|
{
|
|
if(error)
|
|
{
|
|
error->show(this);
|
|
}
|
|
},
|
|
id);
|
|
});
|
|
m_actionDelete->setForceDisabled(true);
|
|
m_toolbar->addAction(m_actionDelete);
|
|
}
|
|
|
|
if(Method* move = object()->getMethod("move"))
|
|
{
|
|
m_actionMoveUp = new MethodAction(Theme::getIcon("up"), *move,
|
|
[this]()
|
|
{
|
|
if(auto* model = m_tableWidget->selectionModel(); model && model->hasSelection())
|
|
{
|
|
if(const auto rows = model->selectedRows(); !rows.empty())
|
|
{
|
|
const int row = rows[0].row();
|
|
callMethod(m_actionMoveUp->method(), nullptr, row, row - 1);
|
|
m_tableWidget->selectRow(row - 1);
|
|
}
|
|
}
|
|
});
|
|
//Override default method name
|
|
m_actionMoveUp->setText(Locale::tr("list:move_up"));
|
|
m_actionMoveUp->setForceDisabled(true);
|
|
|
|
m_actionMoveDown = new MethodAction(Theme::getIcon("down"), *move,
|
|
[this]()
|
|
{
|
|
if(auto* model = m_tableWidget->selectionModel(); model && model->hasSelection())
|
|
{
|
|
if(const auto rows = model->selectedRows(); !rows.empty())
|
|
{
|
|
const int row = rows[0].row();
|
|
callMethod(m_actionMoveDown->method(), nullptr, row, row + 1);
|
|
m_tableWidget->selectRow(row + 1);
|
|
}
|
|
}
|
|
});
|
|
//Override default method name
|
|
m_actionMoveDown->setText(Locale::tr("list:move_down"));
|
|
m_actionMoveDown->setForceDisabled(true);
|
|
}
|
|
|
|
if(Method* method = object()->getMethod("reverse"))
|
|
{
|
|
m_actionReverse = new MethodAction(Theme::getIcon("reverse"), *method);
|
|
}
|
|
|
|
if(Method* method = object()->getMethod("input_monitor"))
|
|
{
|
|
m_actionInputMonitor = new MethodAction(Theme::getIcon("input_monitor"), *method,
|
|
[this]()
|
|
{
|
|
m_requestIdInputMonitor = m_actionInputMonitor->method().call(
|
|
[](const ObjectPtr& inputMonitor, std::optional<const Error>)
|
|
{
|
|
if(inputMonitor)
|
|
MainWindow::instance->showObject(inputMonitor);
|
|
});
|
|
});
|
|
}
|
|
|
|
if(Method* method = object()->getMethod("input_monitor_channel"))
|
|
{
|
|
if(const auto values = method->getAttribute(AttributeName::Values, QVariant()).toList(); !values.isEmpty())
|
|
{
|
|
const QVariantList aliasKeys = method->getAttribute(AttributeName::AliasKeys, QVariant()).toList();
|
|
const QVariantList aliasValues = method->getAttribute(AttributeName::AliasValues, QVariant()).toList();
|
|
|
|
QMenu* menu = new QMenu(this);
|
|
for(const auto& value : values)
|
|
{
|
|
QString text;
|
|
if(int index = aliasKeys.indexOf(value); index != -1)
|
|
text = Locale::instance->parse(aliasValues[index].toString());
|
|
else
|
|
text = value.toString();
|
|
|
|
menu->addAction(text, this,
|
|
[this, channel=value.toUInt()]()
|
|
{
|
|
//cancelRequest(m_requestIdInputMonitor);
|
|
|
|
m_requestIdInputMonitor = callMethodR<ObjectPtr>(m_actionInputMonitorChannel->method(),
|
|
[this](const ObjectPtr& inputMonitor, std::optional<const Error> /*error*/)
|
|
{
|
|
m_requestIdInputMonitor = Connection::invalidRequestId;
|
|
if(inputMonitor)
|
|
MainWindow::instance->showObject(inputMonitor);
|
|
},
|
|
channel);
|
|
});
|
|
}
|
|
|
|
m_actionInputMonitorChannel = new MethodAction(Theme::getIcon("input_monitor"), *method);
|
|
m_actionInputMonitorChannel->setMenu(menu);
|
|
}
|
|
}
|
|
|
|
if(Method* method = object()->getMethod("output_keyboard"))
|
|
{
|
|
const auto values = method->getAttribute(AttributeName::Values, QVariant()).toList();
|
|
if(values.size() > 1)
|
|
{
|
|
QMenu* menu = new QMenu(this);
|
|
for(const auto& value : values)
|
|
{
|
|
menu->addAction(translateEnum(EnumName<OutputChannel>::value, value.toLongLong()), this,
|
|
[this, channel=static_cast<OutputChannel>(value.toUInt())]()
|
|
{
|
|
//cancelRequest(m_requestIdOutputKeyboard);
|
|
|
|
m_requestIdOutputKeyboard = callMethodR<ObjectPtr>(m_actionOutputKeyboard->method(),
|
|
[this](const ObjectPtr& outputKeyboard, std::optional<const Error> /*error*/)
|
|
{
|
|
m_requestIdOutputKeyboard = Connection::invalidRequestId;
|
|
if(outputKeyboard)
|
|
MainWindow::instance->showObject(outputKeyboard);
|
|
},
|
|
channel);
|
|
});
|
|
}
|
|
|
|
m_actionOutputKeyboard = new MethodAction(Theme::getIcon("output_keyboard"), *method);
|
|
m_actionOutputKeyboard->setMenu(menu);
|
|
}
|
|
else if(!values.empty())
|
|
{
|
|
m_actionOutputKeyboard = new MethodAction(Theme::getIcon("output_keyboard"), *method,
|
|
[this, channel=static_cast<OutputChannel>(values.first().toUInt())]()
|
|
{
|
|
//cancelRequest(m_requestIdOutputKeyboard);
|
|
|
|
m_requestIdOutputKeyboard = callMethodR<ObjectPtr>(m_actionOutputKeyboard->method(),
|
|
[this](const ObjectPtr& outputKeyboard, std::optional<const Error> /*error*/)
|
|
{
|
|
m_requestIdOutputKeyboard = Connection::invalidRequestId;
|
|
if(outputKeyboard)
|
|
MainWindow::instance->showObject(outputKeyboard);
|
|
},
|
|
channel);
|
|
});
|
|
}
|
|
}
|
|
|
|
if(m_actionMoveUp || m_actionMoveDown)
|
|
{
|
|
m_toolbar->addSeparator();
|
|
if(m_actionMoveUp)
|
|
m_toolbar->addAction(m_actionMoveUp);
|
|
if(m_actionMoveDown)
|
|
m_toolbar->addAction(m_actionMoveDown);
|
|
}
|
|
|
|
if(m_actionReverse)
|
|
{
|
|
m_toolbar->addSeparator();
|
|
m_toolbar->addAction(m_actionReverse);
|
|
}
|
|
|
|
if(m_actionInputMonitor || m_actionInputMonitorChannel || m_actionOutputKeyboard)
|
|
{
|
|
if(!m_toolbar->actions().empty())
|
|
{
|
|
m_toolbar->addSeparator();
|
|
}
|
|
if(m_actionInputMonitor)
|
|
m_toolbar->addAction(m_actionInputMonitor);
|
|
if(m_actionInputMonitorChannel)
|
|
{
|
|
m_toolbar->addAction(m_actionInputMonitorChannel);
|
|
if(auto* button = qobject_cast<QToolButton*>(m_toolbar->widgetForAction(m_actionInputMonitorChannel)))
|
|
connect(m_actionInputMonitorChannel, &QAction::triggered, button, &QToolButton::showMenu);
|
|
}
|
|
if(m_actionOutputKeyboard)
|
|
{
|
|
m_toolbar->addAction(m_actionOutputKeyboard);
|
|
if(auto* button = qobject_cast<QToolButton*>(m_toolbar->widgetForAction(m_actionOutputKeyboard)))
|
|
connect(m_actionOutputKeyboard, &QAction::triggered, button, &QToolButton::showMenu);
|
|
}
|
|
}
|
|
|
|
{
|
|
QAction* startAll = nullptr;
|
|
if(Method* method = object()->getMethod("start_all"))
|
|
startAll = new MethodAction(Theme::getIcon("run"), *method);
|
|
|
|
QAction* stopAll = nullptr;
|
|
if(Method* method = object()->getMethod("stop_all"))
|
|
stopAll = new MethodAction(Theme::getIcon("stop"), *method);
|
|
|
|
if(startAll || stopAll)
|
|
{
|
|
m_toolbar->addSeparator();
|
|
if(startAll)
|
|
m_toolbar->addAction(startAll);
|
|
if(stopAll)
|
|
m_toolbar->addAction(stopAll);
|
|
}
|
|
}
|
|
|
|
if(auto* method = object()->getMethod("clear_persistent_variables"))
|
|
{
|
|
QWidget* spacer = new QWidget(this);
|
|
spacer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
|
|
spacer->show();
|
|
m_toolbar->addWidget(spacer);
|
|
|
|
m_toolbar->addAction(new MethodAction(Theme::getIcon("clear_persistent_variables"), *method, this));
|
|
}
|
|
|
|
if(!m_toolbar->actions().empty())
|
|
{
|
|
static_cast<QVBoxLayout*>(this->layout())->insertWidget(0, m_toolbar);
|
|
}
|
|
else
|
|
{
|
|
delete m_toolbar;
|
|
m_toolbar = nullptr;
|
|
}
|
|
}
|
|
|
|
ObjectListWidget::~ObjectListWidget()
|
|
{
|
|
object()->connection()->cancelRequest(m_requestIdCreate);
|
|
object()->connection()->cancelRequest(m_requestIdInputMonitor);
|
|
object()->connection()->cancelRequest(m_requestIdOutputKeyboard);
|
|
}
|
|
|
|
void ObjectListWidget::objectDoubleClicked(const QString& id)
|
|
{
|
|
if(hasEdit())
|
|
{
|
|
MainWindow::instance->showObject(id);
|
|
}
|
|
}
|
|
|
|
void ObjectListWidget::tableDoubleClicked(const QModelIndex& index)
|
|
{
|
|
const QString id = m_tableWidget->getRowObjectId(index.row());
|
|
if(!id.isEmpty())
|
|
objectDoubleClicked(id);
|
|
}
|
|
|
|
QStringList ObjectListWidget::getSelectedObjectIds() const
|
|
{
|
|
QStringList ids;
|
|
|
|
if(auto* model = m_tableWidget->selectionModel(); model && model->hasSelection())
|
|
for(const auto& index : model->selectedRows())
|
|
if(QString id = m_tableWidget->getRowObjectId(index.row()); !id.isEmpty())
|
|
ids.append(id);
|
|
|
|
return ids;
|
|
}
|
|
|
|
void ObjectListWidget::objectCreated(const ObjectPtr& object)
|
|
{
|
|
MainWindow::instance->showObject(object);
|
|
}
|
|
|
|
void ObjectListWidget::tableSelectionChanged()
|
|
{
|
|
tableSelectionChanged(m_tableWidget->selectionModel() && m_tableWidget->selectionModel()->hasSelection());
|
|
}
|
|
|
|
void ObjectListWidget::tableSelectionChanged(bool hasSelection)
|
|
{
|
|
if(m_actionEdit)
|
|
m_actionEdit->setEnabled(hasSelection);
|
|
if(m_actionRemove)
|
|
m_actionRemove->setForceDisabled(!hasSelection);
|
|
if(m_actionDelete)
|
|
m_actionDelete->setForceDisabled(!hasSelection);
|
|
if(m_actionMoveUp || m_actionMoveDown)
|
|
{
|
|
const int selectedRow = m_tableWidget->selectionModel() && m_tableWidget->selectionModel()->selectedRows().length() == 1 ? m_tableWidget->selectionModel()->selectedRows()[0].row() : -1;
|
|
if(m_actionMoveUp)
|
|
m_actionMoveUp->setForceDisabled(!hasSelection || selectedRow == 0);
|
|
if(m_actionMoveDown)
|
|
m_actionMoveDown->setForceDisabled(!hasSelection || selectedRow == (m_tableWidget->model()->rowCount() - 1));
|
|
}
|
|
}
|
|
|
|
bool ObjectListWidget::hasEdit() const
|
|
{
|
|
if(object()->classId() == "list.decoder" || object()->classId() == "list.output")
|
|
{
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|