2022-10-09 00:32:02 +04:00
|
|
|
/*
|
|
|
|
* townbuildingswidget.cpp, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
2022-10-16 14:22:34 +04:00
|
|
|
#include "StdInc.h"
|
2024-06-24 03:23:26 +02:00
|
|
|
#include "townbuildingswidget.h"
|
|
|
|
#include "ui_townbuildingswidget.h"
|
2024-07-28 16:07:52 +02:00
|
|
|
#include "mapeditorroles.h"
|
2024-07-21 10:49:40 +00:00
|
|
|
#include "../lib/entities/building/CBuilding.h"
|
|
|
|
#include "../lib/entities/faction/CTownHandler.h"
|
2024-07-20 12:55:17 +00:00
|
|
|
#include "../lib/texts/CGeneralTextHandler.h"
|
2022-09-18 03:23:17 +04:00
|
|
|
|
|
|
|
std::string defaultBuildingIdConversion(BuildingID bId)
|
|
|
|
{
|
|
|
|
switch(bId)
|
|
|
|
{
|
|
|
|
case BuildingID::DEFAULT: return "DEFAULT";
|
|
|
|
case BuildingID::MAGES_GUILD_1: return "MAGES_GUILD_1";
|
|
|
|
case BuildingID::MAGES_GUILD_2: return "MAGES_GUILD_2";
|
|
|
|
case BuildingID::MAGES_GUILD_3: return "MAGES_GUILD_3";
|
|
|
|
case BuildingID::MAGES_GUILD_4: return "MAGES_GUILD_4";
|
|
|
|
case BuildingID::MAGES_GUILD_5: return "MAGES_GUILD_5";
|
|
|
|
case BuildingID::TAVERN: return "TAVERN";
|
|
|
|
case BuildingID::SHIPYARD: return "SHIPYARD";
|
|
|
|
case BuildingID::FORT: return "FORT";
|
|
|
|
case BuildingID::CITADEL: return "CITADEL";
|
|
|
|
case BuildingID::CASTLE: return "CASTLE";
|
|
|
|
case BuildingID::VILLAGE_HALL: return "VILLAGE_HALL";
|
|
|
|
case BuildingID::TOWN_HALL: return "TOWN_HALL";
|
|
|
|
case BuildingID::CITY_HALL: return "CITY_HALL";
|
|
|
|
case BuildingID::CAPITOL: return "CAPITOL";
|
|
|
|
case BuildingID::MARKETPLACE: return "MARKETPLACE";
|
|
|
|
case BuildingID::RESOURCE_SILO: return "RESOURCE_SILO";
|
|
|
|
case BuildingID::BLACKSMITH: return "BLACKSMITH";
|
|
|
|
case BuildingID::SPECIAL_1: return "SPECIAL_1";
|
|
|
|
case BuildingID::SPECIAL_2: return "SPECIAL_2";
|
|
|
|
case BuildingID::SPECIAL_3: return "SPECIAL_3";
|
|
|
|
case BuildingID::SPECIAL_4: return "SPECIAL_4";
|
|
|
|
case BuildingID::HORDE_1: return "HORDE_1";
|
|
|
|
case BuildingID::HORDE_1_UPGR: return "HORDE_1_UPGR";
|
|
|
|
case BuildingID::HORDE_2: return "HORDE_2";
|
|
|
|
case BuildingID::HORDE_2_UPGR: return "HORDE_2_UPGR";
|
|
|
|
case BuildingID::SHIP: return "SHIP";
|
|
|
|
case BuildingID::GRAIL: return "GRAIL";
|
|
|
|
case BuildingID::EXTRA_TOWN_HALL: return "EXTRA_TOWN_HALL";
|
|
|
|
case BuildingID::EXTRA_CITY_HALL: return "EXTRA_CITY_HALL";
|
|
|
|
case BuildingID::EXTRA_CAPITOL: return "EXTRA_CAPITOL";
|
|
|
|
case BuildingID::DWELL_LVL_1: return "DWELL_LVL_1";
|
|
|
|
case BuildingID::DWELL_LVL_2: return "DWELL_LVL_2";
|
|
|
|
case BuildingID::DWELL_LVL_3: return "DWELL_LVL_3";
|
|
|
|
case BuildingID::DWELL_LVL_4: return "DWELL_LVL_4";
|
|
|
|
case BuildingID::DWELL_LVL_5: return "DWELL_LVL_5";
|
|
|
|
case BuildingID::DWELL_LVL_6: return "DWELL_LVL_6";
|
|
|
|
case BuildingID::DWELL_LVL_7: return "DWELL_LVL_7";
|
2024-08-05 21:15:47 +02:00
|
|
|
case BuildingID::DWELL_LVL_8: return "DWELL_LVL_8";
|
2022-09-18 03:23:17 +04:00
|
|
|
case BuildingID::DWELL_LVL_1_UP: return "DWELL_LVL_1_UP";
|
|
|
|
case BuildingID::DWELL_LVL_2_UP: return "DWELL_LVL_2_UP";
|
|
|
|
case BuildingID::DWELL_LVL_3_UP: return "DWELL_LVL_3_UP";
|
|
|
|
case BuildingID::DWELL_LVL_4_UP: return "DWELL_LVL_4_UP";
|
|
|
|
case BuildingID::DWELL_LVL_5_UP: return "DWELL_LVL_5_UP";
|
|
|
|
case BuildingID::DWELL_LVL_6_UP: return "DWELL_LVL_6_UP";
|
|
|
|
case BuildingID::DWELL_LVL_7_UP: return "DWELL_LVL_7_UP";
|
2024-08-05 21:15:47 +02:00
|
|
|
case BuildingID::DWELL_LVL_8_UP: return "DWELL_LVL_8_UP";
|
2022-09-18 03:23:17 +04:00
|
|
|
default:
|
|
|
|
return "UNKNOWN";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-01 23:15:24 +02:00
|
|
|
QStandardItem * getBuildingParentFromTreeModel(const CBuilding * building, const QStandardItemModel & model)
|
2024-07-26 20:42:16 +02:00
|
|
|
{
|
|
|
|
QStandardItem * parent = nullptr;
|
|
|
|
std::vector<QModelIndex> stack(1);
|
2024-08-01 22:28:23 +02:00
|
|
|
do
|
2024-07-26 20:42:16 +02:00
|
|
|
{
|
|
|
|
auto pindex = stack.back();
|
|
|
|
stack.pop_back();
|
|
|
|
auto rowCount = model.rowCount(pindex);
|
|
|
|
for (int i = 0; i < rowCount; ++i)
|
|
|
|
{
|
|
|
|
QModelIndex index = model.index(i, 0, pindex);
|
2024-07-28 16:07:52 +02:00
|
|
|
if (building->upgrade.getNum() == model.itemFromIndex(index)->data(MapEditorRoles::BuildingIDRole).toInt())
|
2024-07-26 20:42:16 +02:00
|
|
|
{
|
|
|
|
parent = model.itemFromIndex(index);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (model.hasChildren(index))
|
|
|
|
stack.push_back(index);
|
|
|
|
}
|
2024-08-01 22:28:23 +02:00
|
|
|
} while(!parent && !stack.empty());
|
2024-07-26 20:42:16 +02:00
|
|
|
return parent;
|
|
|
|
}
|
|
|
|
|
2024-08-01 23:15:24 +02:00
|
|
|
QVariantList getBuildingVariantsFromModel(const QStandardItemModel & model, int modelColumn, Qt::CheckState checkState)
|
2024-07-26 20:42:16 +02:00
|
|
|
{
|
2024-08-01 22:28:23 +02:00
|
|
|
QVariantList result;
|
2024-07-26 20:42:16 +02:00
|
|
|
std::vector<QModelIndex> stack(1);
|
2024-08-01 22:28:23 +02:00
|
|
|
do
|
2024-07-26 20:42:16 +02:00
|
|
|
{
|
|
|
|
auto pindex = stack.back();
|
|
|
|
stack.pop_back();
|
|
|
|
auto rowCount = model.rowCount(pindex);
|
|
|
|
for (int i = 0; i < rowCount; ++i)
|
|
|
|
{
|
|
|
|
QModelIndex index = model.index(i, modelColumn, pindex);
|
2024-08-01 22:28:23 +02:00
|
|
|
auto * item = model.itemFromIndex(index);
|
|
|
|
if(item && item->checkState() == checkState)
|
|
|
|
result.push_back(item->data(MapEditorRoles::BuildingIDRole));
|
2024-07-26 20:42:16 +02:00
|
|
|
index = model.index(i, 0, pindex);
|
|
|
|
if (model.hasChildren(index))
|
|
|
|
stack.push_back(index);
|
|
|
|
}
|
2024-08-01 22:28:23 +02:00
|
|
|
} while(!stack.empty());
|
2024-07-26 20:42:16 +02:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
TownBuildingsWidget::TownBuildingsWidget(CGTownInstance & t, QWidget *parent) :
|
2022-09-18 03:23:17 +04:00
|
|
|
town(t),
|
|
|
|
QDialog(parent),
|
2024-06-24 03:23:26 +02:00
|
|
|
ui(new Ui::TownBuildingsWidget)
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
ui->treeView->setModel(&model);
|
|
|
|
//ui->treeView->setColumnCount(3);
|
2024-07-14 21:00:08 +02:00
|
|
|
model.setHorizontalHeaderLabels(QStringList() << tr("Type") << tr("Enabled") << tr("Built"));
|
2024-06-08 16:33:55 +02:00
|
|
|
connect(&model, &QStandardItemModel::itemChanged, this, &TownBuildingsWidget::onItemChanged);
|
2022-09-18 03:23:17 +04:00
|
|
|
//setAttribute(Qt::WA_DeleteOnClose);
|
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
TownBuildingsWidget::~TownBuildingsWidget()
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
|
|
|
delete ui;
|
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
QStandardItem * TownBuildingsWidget::addBuilding(const CTown & ctown, int bId, std::set<si32> & remaining)
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
|
|
|
BuildingID buildingId(bId);
|
|
|
|
const CBuilding * building = ctown.buildings.at(buildingId);
|
|
|
|
if(!building)
|
|
|
|
{
|
|
|
|
remaining.erase(bId);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2024-07-26 20:42:16 +02:00
|
|
|
QString name = QString::fromStdString(building->getNameTranslated());
|
2022-09-18 03:23:17 +04:00
|
|
|
|
|
|
|
if(name.isEmpty())
|
|
|
|
name = QString::fromStdString(defaultBuildingIdConversion(buildingId));
|
|
|
|
|
|
|
|
QList<QStandardItem *> checks;
|
|
|
|
|
|
|
|
checks << new QStandardItem(name);
|
2024-07-28 16:07:52 +02:00
|
|
|
checks.back()->setData(bId, MapEditorRoles::BuildingIDRole);
|
2022-09-18 03:23:17 +04:00
|
|
|
|
|
|
|
checks << new QStandardItem;
|
|
|
|
checks.back()->setCheckable(true);
|
|
|
|
checks.back()->setCheckState(town.forbiddenBuildings.count(buildingId) ? Qt::Unchecked : Qt::Checked);
|
2024-07-28 16:07:52 +02:00
|
|
|
checks.back()->setData(bId, MapEditorRoles::BuildingIDRole);
|
2022-09-18 03:23:17 +04:00
|
|
|
|
|
|
|
checks << new QStandardItem;
|
|
|
|
checks.back()->setCheckable(true);
|
2024-08-17 22:06:48 +03:00
|
|
|
checks.back()->setCheckState(town.hasBuilt(buildingId) ? Qt::Checked : Qt::Unchecked);
|
2024-07-28 16:07:52 +02:00
|
|
|
checks.back()->setData(bId, MapEditorRoles::BuildingIDRole);
|
2022-09-18 03:23:17 +04:00
|
|
|
|
|
|
|
if(building->getBase() == buildingId)
|
|
|
|
{
|
|
|
|
model.appendRow(checks);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-07-26 20:42:16 +02:00
|
|
|
QStandardItem * parent = getBuildingParentFromTreeModel(building, model);
|
2022-09-18 03:23:17 +04:00
|
|
|
|
|
|
|
if(!parent)
|
|
|
|
parent = addBuilding(ctown, building->upgrade.getNum(), remaining);
|
|
|
|
|
|
|
|
if(!parent)
|
|
|
|
{
|
|
|
|
remaining.erase(bId);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
parent->appendRow(checks);
|
|
|
|
}
|
|
|
|
|
|
|
|
remaining.erase(bId);
|
|
|
|
return checks.front();
|
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
void TownBuildingsWidget::addBuildings(const CTown & ctown)
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
|
|
|
auto buildings = ctown.getAllBuildings();
|
|
|
|
while(!buildings.empty())
|
|
|
|
{
|
|
|
|
addBuilding(ctown, *buildings.begin(), buildings);
|
|
|
|
}
|
|
|
|
ui->treeView->resizeColumnToContents(0);
|
|
|
|
ui->treeView->resizeColumnToContents(1);
|
|
|
|
ui->treeView->resizeColumnToContents(2);
|
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
std::set<BuildingID> TownBuildingsWidget::getBuildingsFromModel(int modelColumn, Qt::CheckState checkState)
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
2024-07-26 20:42:16 +02:00
|
|
|
auto buildingVariants = getBuildingVariantsFromModel(model, modelColumn, checkState);
|
2022-09-18 03:23:17 +04:00
|
|
|
std::set<BuildingID> result;
|
2024-08-01 23:15:24 +02:00
|
|
|
for (const auto & buildingId : buildingVariants)
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
2024-07-26 20:42:16 +02:00
|
|
|
result.insert(buildingId.toInt());
|
2022-09-18 03:23:17 +04:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
std::set<BuildingID> TownBuildingsWidget::getForbiddenBuildings()
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
2024-06-08 16:33:55 +02:00
|
|
|
return getBuildingsFromModel(Column::ENABLED, Qt::Unchecked);
|
2022-10-09 00:32:02 +04:00
|
|
|
}
|
2022-09-18 03:23:17 +04:00
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
std::set<BuildingID> TownBuildingsWidget::getBuiltBuildings()
|
2022-10-09 00:32:02 +04:00
|
|
|
{
|
2024-06-08 16:33:55 +02:00
|
|
|
return getBuildingsFromModel(Column::BUILT, Qt::Checked);
|
2022-09-18 03:23:17 +04:00
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
void TownBuildingsWidget::on_treeView_expanded(const QModelIndex &index)
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
|
|
|
ui->treeView->resizeColumnToContents(0);
|
|
|
|
}
|
|
|
|
|
2024-06-24 03:23:26 +02:00
|
|
|
void TownBuildingsWidget::on_treeView_collapsed(const QModelIndex &index)
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
|
|
|
ui->treeView->resizeColumnToContents(0);
|
|
|
|
}
|
|
|
|
|
2024-06-08 23:13:17 +02:00
|
|
|
void TownBuildingsWidget::on_buildAll_clicked()
|
|
|
|
{
|
|
|
|
setAllRowsColumnCheckState(Column::BUILT, Qt::Checked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TownBuildingsWidget::on_demolishAll_clicked()
|
|
|
|
{
|
|
|
|
setAllRowsColumnCheckState(Column::BUILT, Qt::Unchecked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TownBuildingsWidget::on_enableAll_clicked()
|
|
|
|
{
|
|
|
|
setAllRowsColumnCheckState(Column::ENABLED, Qt::Checked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TownBuildingsWidget::on_disableAll_clicked()
|
|
|
|
{
|
|
|
|
setAllRowsColumnCheckState(Column::ENABLED, Qt::Unchecked);
|
|
|
|
}
|
|
|
|
|
2022-09-18 03:23:17 +04:00
|
|
|
|
2024-08-01 23:15:24 +02:00
|
|
|
void TownBuildingsWidget::setRowColumnCheckState(const QStandardItem * item, Column column, Qt::CheckState checkState) {
|
2024-06-08 16:33:55 +02:00
|
|
|
auto sibling = item->model()->sibling(item->row(), column, item->index());
|
|
|
|
model.itemFromIndex(sibling)->setCheckState(checkState);
|
|
|
|
}
|
|
|
|
|
2024-06-08 23:13:17 +02:00
|
|
|
void TownBuildingsWidget::setAllRowsColumnCheckState(Column column, Qt::CheckState checkState)
|
|
|
|
{
|
2024-07-26 20:42:16 +02:00
|
|
|
std::vector<QModelIndex> stack(1);
|
2024-08-01 22:28:23 +02:00
|
|
|
do
|
2024-06-08 23:13:17 +02:00
|
|
|
{
|
|
|
|
auto parentIndex = stack.back();
|
|
|
|
stack.pop_back();
|
2024-07-26 20:42:16 +02:00
|
|
|
auto rowCount = model.rowCount(parentIndex);
|
|
|
|
for (int i = 0; i < rowCount; ++i)
|
2024-06-08 23:13:17 +02:00
|
|
|
{
|
|
|
|
QModelIndex index = model.index(i, column, parentIndex);
|
|
|
|
if (auto* item = model.itemFromIndex(index))
|
|
|
|
item->setCheckState(checkState);
|
|
|
|
index = model.index(i, 0, parentIndex);
|
|
|
|
if (model.hasChildren(index))
|
|
|
|
stack.push_back(index);
|
|
|
|
}
|
2024-08-01 22:28:23 +02:00
|
|
|
} while(!stack.empty());
|
2024-06-08 23:13:17 +02:00
|
|
|
}
|
|
|
|
|
2024-08-01 23:15:24 +02:00
|
|
|
void TownBuildingsWidget::onItemChanged(const QStandardItem * item) {
|
2024-06-08 16:33:55 +02:00
|
|
|
disconnect(&model, &QStandardItemModel::itemChanged, this, &TownBuildingsWidget::onItemChanged);
|
|
|
|
auto rowFirstColumnIndex = item->model()->sibling(item->row(), Column::TYPE, item->index());
|
|
|
|
QStandardItem * nextRow = model.itemFromIndex(rowFirstColumnIndex);
|
|
|
|
if (item->checkState() == Qt::Checked) {
|
|
|
|
while (nextRow) {
|
|
|
|
setRowColumnCheckState(nextRow, Column(item->column()), Qt::Checked);
|
|
|
|
if (item->column() == Column::BUILT) {
|
|
|
|
setRowColumnCheckState(nextRow, Column::ENABLED, Qt::Checked);
|
|
|
|
}
|
|
|
|
nextRow = nextRow->parent();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (item->checkState() == Qt::Unchecked) {
|
2024-06-08 23:32:58 +02:00
|
|
|
std::vector<QStandardItem*> stack;
|
|
|
|
stack.push_back(nextRow);
|
2024-08-01 22:28:23 +02:00
|
|
|
do
|
|
|
|
{
|
2024-06-08 23:32:58 +02:00
|
|
|
nextRow = stack.back();
|
|
|
|
stack.pop_back();
|
2024-06-08 16:33:55 +02:00
|
|
|
setRowColumnCheckState(nextRow, Column(item->column()), Qt::Unchecked);
|
|
|
|
if (item->column() == Column::ENABLED) {
|
|
|
|
setRowColumnCheckState(nextRow, Column::BUILT, Qt::Unchecked);
|
|
|
|
}
|
2024-06-08 23:32:58 +02:00
|
|
|
if (nextRow->hasChildren()) {
|
|
|
|
for (int i = 0; i < nextRow->rowCount(); ++i) {
|
|
|
|
stack.push_back(nextRow->child(i, Column::TYPE));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-01 22:28:23 +02:00
|
|
|
} while(!stack.empty());
|
2024-06-08 16:33:55 +02:00
|
|
|
}
|
|
|
|
connect(&model, &QStandardItemModel::itemChanged, this, &TownBuildingsWidget::onItemChanged);
|
|
|
|
}
|
|
|
|
|
2022-09-18 03:23:17 +04:00
|
|
|
TownBuildingsDelegate::TownBuildingsDelegate(CGTownInstance & t): town(t), QStyledItemDelegate()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
QWidget * TownBuildingsDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
|
|
{
|
2024-06-24 03:23:26 +02:00
|
|
|
return new TownBuildingsWidget(town, parent);
|
2022-09-18 03:23:17 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TownBuildingsDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const
|
|
|
|
{
|
2024-06-24 03:23:26 +02:00
|
|
|
if(auto * ed = qobject_cast<TownBuildingsWidget *>(editor))
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
2024-10-05 19:37:52 +00:00
|
|
|
auto * ctown = town.getTown();
|
2022-09-18 03:23:17 +04:00
|
|
|
if(!ctown)
|
|
|
|
ctown = VLC->townh->randomTown;
|
|
|
|
if(!ctown)
|
|
|
|
throw std::runtime_error("No Town defined for type selected");
|
|
|
|
|
|
|
|
ed->addBuildings(*ctown);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QStyledItemDelegate::setEditorData(editor, index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TownBuildingsDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const
|
|
|
|
{
|
2024-06-24 03:23:26 +02:00
|
|
|
if(auto * ed = qobject_cast<TownBuildingsWidget *>(editor))
|
2022-09-18 03:23:17 +04:00
|
|
|
{
|
|
|
|
town.forbiddenBuildings = ed->getForbiddenBuildings();
|
2024-08-17 22:06:48 +03:00
|
|
|
town.removeAllBuildings();
|
|
|
|
for(const auto & building : ed->getBuiltBuildings())
|
|
|
|
town.addBuilding(building);
|
2022-09-18 03:23:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QStyledItemDelegate::setModelData(editor, model, index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|