templateeditor.cpp 36 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064
  1. /*
  2. * templateeditor.cpp, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #include "StdInc.h"
  11. #include "templateeditor.h"
  12. #include "ui_templateeditor.h"
  13. #include "graphicelements/CardItem.h"
  14. #include "graphicelements/LineItem.h"
  15. #include "terrainselector.h"
  16. #include "factionselector.h"
  17. #include "mineselector.h"
  18. #include "GeometryAlgorithm.h"
  19. #include "../helper.h"
  20. #include "../../lib/VCMIDirs.h"
  21. #include "../../lib/rmg/CRmgTemplate.h"
  22. #include "../../lib/texts/MetaString.h"
  23. TemplateEditor::TemplateEditor():
  24. ui(new Ui::TemplateEditor)
  25. {
  26. ui->setupUi(this);
  27. setWindowIcon(QIcon{":/icons/menu-game.png"});
  28. ui->actionOpen->setIcon(QIcon{":/icons/document-open.png"});
  29. ui->actionSave->setIcon(QIcon{":/icons/document-save.png"});
  30. ui->actionNew->setIcon(QIcon{":/icons/document-new.png"});
  31. ui->actionAddZone->setIcon(QIcon{":/icons/zone-add.svg"});
  32. ui->actionRemoveZone->setIcon(QIcon{":/icons/zone-remove.svg"});
  33. ui->actionAutoPosition->setIcon(QIcon{":/icons/zones-layout.svg"});
  34. ui->actionZoom_in->setIcon(QIcon{":/icons/zoom_plus.png"});
  35. ui->actionZoom_out->setIcon(QIcon{":/icons/zoom_minus.png"});
  36. ui->actionZoom_auto->setIcon(QIcon{":/icons/zoom_base.png"});
  37. ui->actionZoom_reset->setIcon(QIcon{":/icons/zoom_zero.png"});
  38. templateScene.reset(new TemplateScene());
  39. ui->templateView->setScene(templateScene.get());
  40. ui->toolBox->setVisible(false);
  41. ui->spinBoxZoneVisPosX->setSingleStep(CardItem::GRID_SIZE);
  42. ui->spinBoxZoneVisPosY->setSingleStep(CardItem::GRID_SIZE);
  43. ui->templateView->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
  44. ui->templateView->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
  45. ui->templateView->setSceneRect(-3000, -3000, 6000, 6000);
  46. ui->templateView->centerOn(QPointF(0, 0));
  47. ui->templateView->setDragMode(QGraphicsView::DragMode::ScrollHandDrag);
  48. loadContent();
  49. setTitle();
  50. setWindowModality(Qt::ApplicationModal);
  51. show();
  52. }
  53. TemplateEditor::~TemplateEditor()
  54. {
  55. delete ui;
  56. }
  57. void TemplateEditor::initContent()
  58. {
  59. ui->comboBoxTemplateSelection->clear();
  60. ui->toolBox->setVisible(false);
  61. ui->pageZone->setEnabled(false);
  62. if(templates.empty())
  63. return;
  64. ui->toolBox->setVisible(true);
  65. selectedTemplate = templates.begin()->first;
  66. for(auto & tpl : templates)
  67. ui->comboBoxTemplateSelection->addItem(QString::fromStdString(tpl.first));
  68. }
  69. void TemplateEditor::autoPositionZones()
  70. {
  71. auto & zones = templates[selectedTemplate]->getZones();
  72. std::vector<GeometryAlgorithm::Node> nodes;
  73. std::default_random_engine rng(0);
  74. std::uniform_real_distribution<double> distX(0.0, 500);
  75. std::uniform_real_distribution<double> distY(0.0, 500);
  76. for(auto & item : zones)
  77. {
  78. GeometryAlgorithm::Node node;
  79. node.x = distX(rng);
  80. node.y = distY(rng);
  81. node.id = item.first;
  82. nodes.push_back(node);
  83. }
  84. std::vector<GeometryAlgorithm::Edge> edges;
  85. for(auto & item : templates[selectedTemplate]->getConnectedZoneIds())
  86. edges.push_back({
  87. vstd::find_pos_if(nodes, [item](auto & elem){ return elem.id == item.getZoneA(); }),
  88. vstd::find_pos_if(nodes, [item](auto & elem){ return elem.id == item.getZoneB(); })
  89. });
  90. GeometryAlgorithm::forceDirectedLayout(nodes, edges, 1000, 500, 500);
  91. for(auto & item : nodes)
  92. zones.at(item.id)->setVisiblePosition(Point(CardItem::GRID_SIZE * round(item.x / CardItem::GRID_SIZE), CardItem::GRID_SIZE * round(item.y / CardItem::GRID_SIZE)));
  93. }
  94. void TemplateEditor::loadContent(bool autoPosition)
  95. {
  96. cards.clear();
  97. lines.clear();
  98. selectedZone = -1;
  99. templateScene->clear();
  100. if(templates.empty())
  101. return;
  102. auto & zones = templates[selectedTemplate]->getZones();
  103. if(autoPosition || std::all_of(zones.begin(), zones.end(), [](auto & item){ return item.second->getVisiblePosition().x == 0 && item.second->getVisiblePosition().y == 0; }))
  104. autoPositionZones();
  105. for(auto & zone : zones)
  106. {
  107. auto svgItem = new CardItem();
  108. svgItem->setSelectCallback([this, svgItem](bool selected){
  109. ui->pageZone->setDisabled(!selected);
  110. ui->toolBox->setCurrentIndex(selected ? 1 : 0);
  111. ui->actionRemoveZone->setEnabled(selected);
  112. if(selected)
  113. {
  114. selectedZone = svgItem->getId();
  115. loadZoneMenuContent();
  116. }
  117. else
  118. selectedZone = -1;
  119. });
  120. svgItem->setPosChangeCallback([this, svgItem](QPointF pos){
  121. updateConnectionLines();
  122. for(auto & zone : templates[selectedTemplate]->getZones())
  123. if(zone.first == svgItem->getId())
  124. {
  125. zone.second->setVisiblePosition(Point(svgItem->pos().toPoint().rx() + (svgItem->boundingRect().width() * svgItem->scale() / 2), svgItem->pos().toPoint().ry() + (svgItem->boundingRect().height() * svgItem->scale() / 2)));
  126. zone.second->setVisibleSize(svgItem->scale());
  127. }
  128. loadZoneMenuContent(true);
  129. });
  130. svgItem->setFlags(QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsSelectable | QGraphicsItem::ItemSendsGeometryChanges);
  131. templateScene->addItem(svgItem);
  132. cards[zone.first] = svgItem;
  133. updateZoneCards();
  134. }
  135. updateConnectionLines(true);
  136. updateZonePositions();
  137. ui->templateView->show();
  138. ui->toolBox->setCurrentIndex(0);
  139. connect(ui->toolBox, &QToolBox::currentChanged, this, [this](int index) mutable {
  140. if(!ui->pageZone->isEnabled() && index == 1)
  141. ui->toolBox->setCurrentIndex(0);
  142. });
  143. ui->lineEditName->setText(QString::fromStdString(templates[selectedTemplate]->getName()));
  144. ui->lineEditDescription->setText(QString::fromStdString(templates[selectedTemplate]->getDescription()));
  145. ui->spinBoxMinSizeX->setValue(templates[selectedTemplate]->getMapSizes().first.x);
  146. ui->spinBoxMinSizeY->setValue(templates[selectedTemplate]->getMapSizes().first.y);
  147. ui->spinBoxMinSizeZ->setValue(templates[selectedTemplate]->getMapSizes().first.z);
  148. ui->spinBoxMaxSizeX->setValue(templates[selectedTemplate]->getMapSizes().second.x);
  149. ui->spinBoxMaxSizeY->setValue(templates[selectedTemplate]->getMapSizes().second.y);
  150. ui->spinBoxMaxSizeZ->setValue(templates[selectedTemplate]->getMapSizes().second.z);
  151. ui->checkBoxAllowedWaterContentNone->setChecked(templates[selectedTemplate]->allowedWaterContent.count(EWaterContent::EWaterContent::NONE));
  152. ui->checkBoxAllowedWaterContentNormal->setChecked(templates[selectedTemplate]->allowedWaterContent.count(EWaterContent::EWaterContent::NORMAL));
  153. ui->checkBoxAllowedWaterContentIslands->setChecked(templates[selectedTemplate]->allowedWaterContent.count(EWaterContent::EWaterContent::ISLANDS));
  154. auto setPlayersTable = [this](QTableWidget * widget, std::vector<std::pair<int, int>> & range){
  155. widget->clear();
  156. widget->clearContents();
  157. widget->setRowCount(0);
  158. widget->setColumnCount(3);
  159. widget->setHorizontalHeaderLabels({ tr("Min"), tr("Max"), tr("Action") });
  160. widget->horizontalHeader()->setStretchLastSection(true);
  161. for(int i = 0; i < range.size(); i++)
  162. {
  163. widget->insertRow(i);
  164. QSpinBox* minSpin = new QSpinBox();
  165. QSpinBox* maxSpin = new QSpinBox();
  166. QPushButton* delButton = new QPushButton();
  167. minSpin->setRange(0, 999);
  168. maxSpin->setRange(0, 999);
  169. minSpin->setValue(range.at(i).first);
  170. maxSpin->setValue(range.at(i).second);
  171. connect(minSpin, static_cast<void(QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, [i, &range](int val){
  172. range.at(i).first = val;
  173. });
  174. connect(maxSpin, static_cast<void(QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, [i, &range](int val){
  175. range.at(i).second = val;
  176. });
  177. delButton->setText(tr("Delete"));
  178. connect(delButton, &QPushButton::clicked, this, [this, i, &range](){
  179. range.erase(range.begin() + i);
  180. loadContent();
  181. });
  182. widget->setCellWidget(i, 0, minSpin);
  183. widget->setCellWidget(i, 1, maxSpin);
  184. widget->setCellWidget(i, 2, delButton);
  185. }
  186. widget->insertRow(widget->rowCount());
  187. QPushButton* addButton = new QPushButton();
  188. addButton->setText(tr("Add"));
  189. connect(addButton, &QPushButton::clicked, this, [this, &range](){
  190. range.push_back(std::make_pair(0, 0));
  191. loadContent();
  192. });
  193. widget->setCellWidget(widget->rowCount() - 1, 2, addButton);
  194. };
  195. setPlayersTable(ui->tableWidgetPlayersPlayer, templates[selectedTemplate]->players.range);
  196. setPlayersTable(ui->tableWidgetPlayersHuman, templates[selectedTemplate]->humanPlayers.range);
  197. loadZoneConnectionMenuContent();
  198. }
  199. void TemplateEditor::updateZonePositions()
  200. {
  201. for(auto & card : cards)
  202. {
  203. auto & zone = templates[selectedTemplate]->getZones().at(card.first);
  204. card.second->setPos(zone->getVisiblePosition().x - (card.second->boundingRect().width() * card.second->scale() / 2), zone->getVisiblePosition().y - (card.second->boundingRect().height() * card.second->scale() / 2));
  205. card.second->setScale(zone->getVisibleSize());
  206. }
  207. updateConnectionLines();
  208. }
  209. QString TemplateEditor::getZoneToolTip(std::shared_ptr<rmg::ZoneOptions> zone)
  210. {
  211. QString tmp{};
  212. tmp.append(tr("ID: %1").arg(QString::number(zone->getId())));
  213. tmp.append("\n");
  214. tmp.append(tr("Max treasure: %1").arg(QString::number(zone->getMaxTreasureValue())));
  215. //TODO: extend other interesting things (terrains, towns, ...)
  216. return tmp;
  217. }
  218. void TemplateEditor::updateZoneCards(TRmgTemplateZoneId id)
  219. {
  220. for(auto & card : cards)
  221. {
  222. if(card.first != id && id > -1)
  223. continue;
  224. auto & zone = templates[selectedTemplate]->getZones().at(card.first);
  225. auto type = zone->getType();
  226. if(type == ETemplateZoneType::PLAYER_START || type == ETemplateZoneType::CPU_START)
  227. card.second->setPlayerColor(PlayerColor(*zone->getOwner()));
  228. else if(type == ETemplateZoneType::TREASURE)
  229. card.second->setMultiFillColor(QColor(165, 125, 55), QColor(250, 229, 157));
  230. else if(type == ETemplateZoneType::WATER)
  231. card.second->setMultiFillColor(QColor(55, 68, 165), QColor(157, 250, 247));
  232. else
  233. card.second->setFillColor(QColor(200, 200, 200));
  234. card.second->setJunction(type == ETemplateZoneType::JUNCTION);
  235. card.second->setId(card.first);
  236. for(auto & res : {GameResID::WOOD, GameResID::ORE, GameResID::MERCURY, GameResID::SULFUR, GameResID::CRYSTAL, GameResID::GEMS, GameResID::GOLD})
  237. card.second->setResAmount(res, zone->getMinesInfo().count(res) ? zone->getMinesInfo().at(res) : 0);
  238. card.second->setChestValue(zone->getMaxTreasureValue());
  239. card.second->setMonsterStrength(zone->monsterStrength);
  240. card.second->setToolTip(getZoneToolTip(zone));
  241. card.second->setScale(zone->getVisibleSize());
  242. card.second->updateContent();
  243. }
  244. }
  245. void TemplateEditor::loadZoneMenuContent(bool onlyPosition)
  246. {
  247. if(selectedZone < 0 || selectedTemplate.empty())
  248. return;
  249. auto setValue = [](auto& target, const auto& newValue){ target->setValue(newValue); };
  250. auto & zone = templates[selectedTemplate]->getZones().at(selectedZone);
  251. setValue(ui->spinBoxZoneVisPosX, zone->getVisiblePosition().x);
  252. setValue(ui->spinBoxZoneVisPosY, zone->getVisiblePosition().y);
  253. setValue(ui->doubleSpinBoxZoneVisSize, zone->getVisibleSize());
  254. if(onlyPosition)
  255. return;
  256. setValue(ui->spinBoxZoneSize, zone->getSize());
  257. setValue(ui->spinBoxTownCountPlayer, zone->playerTowns.townCount);
  258. setValue(ui->spinBoxCastleCountPlayer, zone->playerTowns.castleCount);
  259. setValue(ui->spinBoxTownDensityPlayer, zone->playerTowns.townDensity);
  260. setValue(ui->spinBoxCastleDensityPlayer, zone->playerTowns.castleDensity);
  261. setValue(ui->spinBoxTownCountNeutral, zone->neutralTowns.townCount);
  262. setValue(ui->spinBoxCastleCountNeutral, zone->neutralTowns.castleCount);
  263. setValue(ui->spinBoxTownDensityNeutral, zone->neutralTowns.townDensity);
  264. setValue(ui->spinBoxCastleDensityNeutral, zone->neutralTowns.castleDensity);
  265. ui->checkBoxMatchTerrainToTown->setChecked(zone->matchTerrainToTown);
  266. ui->checkBoxTownsAreSameType->setChecked(zone->townsAreSameType);
  267. ui->checkBoxZoneLinkTowns->setChecked(zone->townsLikeZone != rmg::ZoneOptions::NO_ZONE);
  268. ui->checkBoxZoneLinkMines->setChecked(zone->minesLikeZone != rmg::ZoneOptions::NO_ZONE);
  269. ui->checkBoxZoneLinkTerrain->setChecked(zone->terrainTypeLikeZone != rmg::ZoneOptions::NO_ZONE);
  270. ui->checkBoxZoneLinkTreasure->setChecked(zone->treasureLikeZone != rmg::ZoneOptions::NO_ZONE);
  271. ui->checkBoxZoneLinkCustomObjects->setChecked(zone->customObjectsLikeZone != rmg::ZoneOptions::NO_ZONE);
  272. setValue(ui->spinBoxZoneLinkTowns, zone->townsLikeZone != rmg::ZoneOptions::NO_ZONE ? zone->townsLikeZone : 0);
  273. setValue(ui->spinBoxZoneLinkMines, zone->minesLikeZone != rmg::ZoneOptions::NO_ZONE ? zone->minesLikeZone : 0);
  274. setValue(ui->spinBoxZoneLinkTerrain, zone->terrainTypeLikeZone != rmg::ZoneOptions::NO_ZONE ? zone->terrainTypeLikeZone : 0);
  275. setValue(ui->spinBoxZoneLinkTreasure, zone->treasureLikeZone != rmg::ZoneOptions::NO_ZONE ? zone->treasureLikeZone : 0);
  276. setValue(ui->spinBoxZoneLinkCustomObjects, zone->customObjectsLikeZone != rmg::ZoneOptions::NO_ZONE ? zone->customObjectsLikeZone : 0);
  277. ui->spinBoxZoneLinkTowns->setEnabled(zone->townsLikeZone != rmg::ZoneOptions::NO_ZONE);
  278. ui->spinBoxZoneLinkMines->setEnabled(zone->minesLikeZone != rmg::ZoneOptions::NO_ZONE);
  279. ui->spinBoxZoneLinkTerrain->setEnabled(zone->terrainTypeLikeZone != rmg::ZoneOptions::NO_ZONE);
  280. ui->spinBoxZoneLinkTreasure->setEnabled(zone->treasureLikeZone != rmg::ZoneOptions::NO_ZONE);
  281. ui->spinBoxZoneLinkCustomObjects->setEnabled(zone->customObjectsLikeZone != rmg::ZoneOptions::NO_ZONE);
  282. setValue(ui->spinBoxZoneId, zone->id);
  283. ui->spinBoxZoneId->setEnabled(false);
  284. ui->comboBoxZoneType->clear();
  285. ui->comboBoxZoneType->addItem(tr("Player start"), QVariant(static_cast<int>(ETemplateZoneType::PLAYER_START)));
  286. ui->comboBoxZoneType->addItem(tr("CPU start"), QVariant(static_cast<int>(ETemplateZoneType::CPU_START)));
  287. ui->comboBoxZoneType->addItem(tr("Treasure"), QVariant(static_cast<int>(ETemplateZoneType::TREASURE)));
  288. ui->comboBoxZoneType->addItem(tr("Junction"), QVariant(static_cast<int>(ETemplateZoneType::JUNCTION)));
  289. ui->comboBoxZoneType->addItem(tr("Water"), QVariant(static_cast<int>(ETemplateZoneType::WATER)));
  290. ui->comboBoxZoneType->addItem(tr("Sealed"), QVariant(static_cast<int>(ETemplateZoneType::SEALED)));
  291. for (int i = 0; i < ui->comboBoxZoneType->count(); ++i)
  292. if (ui->comboBoxZoneType->itemData(i).toInt() == static_cast<int>(zone->getType()))
  293. ui->comboBoxZoneType->setCurrentIndex(i);
  294. ui->comboBoxZoneOwner->clear();
  295. auto type = static_cast<ETemplateZoneType>(ui->comboBoxZoneType->currentData().toInt());
  296. if((type == ETemplateZoneType::PLAYER_START || type == ETemplateZoneType::CPU_START))
  297. {
  298. ui->comboBoxZoneOwner->setEnabled(true);
  299. for(auto color = PlayerColor(0); color < PlayerColor::PLAYER_LIMIT; ++color)
  300. {
  301. MetaString str;
  302. str.appendName(color);
  303. ui->comboBoxZoneOwner->addItem(QString::fromStdString(str.toString()), QVariant(static_cast<int>(color)));
  304. }
  305. for (int i = 0; i < ui->comboBoxZoneOwner->count(); ++i)
  306. if (ui->comboBoxZoneOwner->itemData(i).toInt() == static_cast<int>(*zone->getOwner()))
  307. ui->comboBoxZoneOwner->setCurrentIndex(i);
  308. }
  309. else
  310. {
  311. ui->comboBoxZoneOwner->addItem(tr("None"));
  312. ui->comboBoxZoneOwner->setEnabled(false);
  313. }
  314. ui->comboBoxMonsterStrength->clear();
  315. ui->comboBoxMonsterStrength->addItem(tr("None"), QVariant(static_cast<int>(EMonsterStrength::EMonsterStrength::ZONE_NONE)));
  316. ui->comboBoxMonsterStrength->addItem(tr("Random"), QVariant(static_cast<int>(EMonsterStrength::EMonsterStrength::RANDOM)));
  317. ui->comboBoxMonsterStrength->addItem(tr("Weak"), QVariant(static_cast<int>(EMonsterStrength::EMonsterStrength::ZONE_WEAK)));
  318. ui->comboBoxMonsterStrength->addItem(tr("Normal"), QVariant(static_cast<int>(EMonsterStrength::EMonsterStrength::ZONE_NORMAL)));
  319. ui->comboBoxMonsterStrength->addItem(tr("Strong"), QVariant(static_cast<int>(EMonsterStrength::EMonsterStrength::ZONE_STRONG)));
  320. for (int i = 0; i < ui->comboBoxMonsterStrength->count(); ++i)
  321. if (ui->comboBoxMonsterStrength->itemData(i).toInt() == static_cast<int>(zone->monsterStrength))
  322. ui->comboBoxMonsterStrength->setCurrentIndex(i);
  323. }
  324. void TemplateEditor::loadZoneConnectionMenuContent()
  325. {
  326. auto widget = ui->tableWidgetConnections;
  327. auto & connections = templates[selectedTemplate]->connections;
  328. widget->clear();
  329. widget->clearContents();
  330. widget->setRowCount(0);
  331. widget->setColumnCount(6);
  332. widget->setHorizontalHeaderLabels({ tr("Zone A"), tr("Zone B"), tr("Guard"), tr("Type"), tr("Road"), "" });
  333. for (int i = 0; i < connections.size(); i++)
  334. {
  335. int row = widget->rowCount();
  336. widget->insertRow(row);
  337. QSpinBox* zoneA = new QSpinBox();
  338. QSpinBox* zoneB = new QSpinBox();
  339. QSpinBox* guardStrength = new QSpinBox();
  340. QComboBox* connectionType = new QComboBox();
  341. QComboBox* hasRoad = new QComboBox();
  342. QPushButton* delButton = new QPushButton();
  343. zoneA->setRange(0, 999);
  344. zoneB->setRange(0, 999);
  345. guardStrength->setRange(0, 99999);
  346. connectionType->addItem(tr("Guarded"), QVariant(static_cast<int>(rmg::EConnectionType::GUARDED)));
  347. connectionType->addItem(tr("Fictive"), QVariant(static_cast<int>(rmg::EConnectionType::FICTIVE)));
  348. connectionType->addItem(tr("Repulsive"), QVariant(static_cast<int>(rmg::EConnectionType::REPULSIVE)));
  349. connectionType->addItem(tr("Wide"), QVariant(static_cast<int>(rmg::EConnectionType::WIDE)));
  350. connectionType->addItem(tr("Force portal"), QVariant(static_cast<int>(rmg::EConnectionType::FORCE_PORTAL)));
  351. hasRoad->addItem(tr("Random"), QVariant(static_cast<int>(rmg::ERoadOption::ROAD_RANDOM)));
  352. hasRoad->addItem(tr("Yes"), QVariant(static_cast<int>(rmg::ERoadOption::ROAD_TRUE)));
  353. hasRoad->addItem(tr("No"), QVariant(static_cast<int>(rmg::ERoadOption::ROAD_FALSE)));
  354. zoneA->setValue(connections.at(i).getZoneA());
  355. zoneB->setValue(connections.at(i).getZoneB());
  356. guardStrength->setValue(connections.at(i).getGuardStrength());
  357. for (int j = 0; j < connectionType->count(); ++j)
  358. if (connectionType->itemData(j).toInt() == static_cast<int>(connections[i].getConnectionType()))
  359. connectionType->setCurrentIndex(j);
  360. for (int j = 0; j < hasRoad->count(); ++j)
  361. if (hasRoad->itemData(j).toInt() == static_cast<int>(connections[i].hasRoad))
  362. hasRoad->setCurrentIndex(j);
  363. delButton->setText(tr("Del"));
  364. delButton->setToolTip(tr("Delete"));
  365. connect(zoneA, static_cast<void(QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, [this, i, &connections](int val){
  366. connections.at(i).zoneA = val;
  367. updateConnectionLines(true);
  368. changed();
  369. });
  370. connect(zoneB, static_cast<void(QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, [this, i, &connections](int val){
  371. connections.at(i).zoneB = val;
  372. updateConnectionLines(true);
  373. changed();
  374. });
  375. connect(guardStrength, static_cast<void(QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, [this, i, &connections](int val){
  376. connections.at(i).guardStrength = val;
  377. updateConnectionLines();
  378. changed();
  379. });
  380. connect(connectionType, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [this, i, &connections, connectionType](int val){
  381. connections.at(i).connectionType = static_cast<rmg::EConnectionType>(connectionType->itemData(val).toInt());
  382. updateConnectionLines();
  383. changed();
  384. });
  385. connect(hasRoad, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [this, i, &connections, hasRoad](int val){
  386. connections.at(i).hasRoad = static_cast<rmg::ERoadOption>(hasRoad->itemData(val).toInt());
  387. updateConnectionLines();
  388. changed();
  389. });
  390. connect(delButton, &QPushButton::pressed, this, [this, i, &connections](){
  391. connections.erase(connections.begin() + i);
  392. updateConnectionLines(true);
  393. loadZoneConnectionMenuContent();
  394. changed();
  395. });
  396. widget->setCellWidget(i, 0, zoneA);
  397. widget->setCellWidget(i, 1, zoneB);
  398. widget->setCellWidget(i, 2, guardStrength);
  399. widget->setCellWidget(i, 3, connectionType);
  400. widget->setCellWidget(i, 4, hasRoad);
  401. widget->setCellWidget(i, 5, delButton);
  402. };
  403. widget->resizeColumnsToContents();
  404. }
  405. void TemplateEditor::saveZoneMenuContent()
  406. {
  407. if(selectedZone < 0 || selectedTemplate.empty())
  408. return;
  409. auto zone = templates[selectedTemplate]->getZones().at(selectedZone);
  410. auto type = static_cast<ETemplateZoneType>(ui->comboBoxZoneType->currentData().toInt());
  411. zone->setVisiblePosition(Point(ui->spinBoxZoneVisPosX->value(), ui->spinBoxZoneVisPosY->value()));
  412. zone->setVisibleSize(ui->doubleSpinBoxZoneVisSize->value());
  413. zone->setType(type);
  414. zone->setSize(ui->spinBoxZoneSize->value());
  415. zone->playerTowns.townCount = ui->spinBoxTownCountPlayer->value();
  416. zone->playerTowns.castleCount = ui->spinBoxCastleCountPlayer->value();
  417. zone->playerTowns.townDensity = ui->spinBoxTownDensityPlayer->value();
  418. zone->playerTowns.castleDensity = ui->spinBoxCastleDensityPlayer->value();
  419. zone->neutralTowns.townCount = ui->spinBoxTownCountNeutral->value();
  420. zone->neutralTowns.castleCount = ui->spinBoxCastleCountNeutral->value();
  421. zone->neutralTowns.townDensity = ui->spinBoxTownDensityNeutral->value();
  422. zone->neutralTowns.castleDensity = ui->spinBoxCastleDensityNeutral->value();
  423. zone->matchTerrainToTown = ui->checkBoxMatchTerrainToTown->isChecked();
  424. zone->townsAreSameType = ui->checkBoxTownsAreSameType->isChecked();
  425. zone->monsterStrength = static_cast<EMonsterStrength::EMonsterStrength>(ui->comboBoxMonsterStrength->currentData().toInt());
  426. zone->id = ui->spinBoxZoneId->value();
  427. zone->townsLikeZone = ui->checkBoxZoneLinkTowns->isChecked() ? ui->spinBoxZoneLinkTowns->value() : rmg::ZoneOptions::NO_ZONE;
  428. zone->minesLikeZone = ui->checkBoxZoneLinkMines->isChecked() ? ui->spinBoxZoneLinkMines->value() : rmg::ZoneOptions::NO_ZONE;
  429. zone->terrainTypeLikeZone = ui->checkBoxZoneLinkTerrain->isChecked() ? ui->spinBoxZoneLinkTerrain->value() : rmg::ZoneOptions::NO_ZONE;
  430. zone->treasureLikeZone = ui->checkBoxZoneLinkTreasure->isChecked() ? ui->spinBoxZoneLinkTreasure->value() : rmg::ZoneOptions::NO_ZONE;
  431. zone->customObjectsLikeZone = ui->checkBoxZoneLinkCustomObjects->isChecked() ? ui->spinBoxZoneLinkCustomObjects->value() : rmg::ZoneOptions::NO_ZONE;
  432. if((type == ETemplateZoneType::PLAYER_START || type == ETemplateZoneType::CPU_START))
  433. zone->owner = std::optional<int>(static_cast<PlayerColor>(ui->comboBoxZoneOwner->currentData().toInt()));
  434. else
  435. zone->owner = std::nullopt;
  436. updateZonePositions();
  437. updateZoneCards(selectedZone);
  438. changed();
  439. }
  440. void TemplateEditor::updateConnectionLines(bool recreate)
  441. {
  442. if(recreate)
  443. {
  444. for(auto & line : lines)
  445. templateScene->removeItem(line);
  446. lines.clear();
  447. for(int i = 0; i < templates[selectedTemplate]->getConnectedZoneIds().size(); i++)
  448. {
  449. auto & connection = templates[selectedTemplate]->getConnectedZoneIds()[i];
  450. auto line = new LineItem();
  451. line->setId(i);
  452. line->setLineToolTip(tr("Zone A: %1\nZone B: %2\nGuard: %3").arg(QString::number(connection.zoneA)).arg(QString::number(connection.zoneB)).arg(QString::number(connection.guardStrength)));
  453. line->setClickCallback([this, line](){
  454. ui->toolBox->setCurrentIndex(2);
  455. ui->tableWidgetConnections->selectRow(line->getId());
  456. });
  457. lines.push_back(line);
  458. templateScene->addItem(line);
  459. }
  460. }
  461. const auto& connections = templates[selectedTemplate]->getConnectedZoneIds();
  462. for (size_t i = 0; i < connections.size(); ++i)
  463. {
  464. auto& connection = connections[i];
  465. if(!cards.count(connection.getZoneA()) || !cards.count(connection.getZoneB()) || i >= lines.size())
  466. continue;
  467. auto getCenter = [](auto & i){
  468. return i->pos() + QPointF(i->boundingRect().width() / 2, i->boundingRect().height() / 2) * i->scale();
  469. };
  470. auto posA = getCenter(cards[connection.getZoneA()]);
  471. auto posB = getCenter(cards[connection.getZoneB()]);
  472. lines[i]->setLine(QLineF(posA.x(), posA.y(), posB.x(), posB.y()));
  473. lines[i]->setText(QString::number(connection.getGuardStrength()));
  474. std::map<rmg::EConnectionType, QPen> pens = {
  475. {rmg::EConnectionType::GUARDED, QPen(Qt::black, 5, Qt::SolidLine, Qt::SquareCap, Qt::BevelJoin)},
  476. {rmg::EConnectionType::FICTIVE, QPen(Qt::black, 5, Qt::DashLine, Qt::SquareCap, Qt::BevelJoin)},
  477. {rmg::EConnectionType::REPULSIVE, QPen(Qt::black, 5, Qt::DashDotLine, Qt::SquareCap, Qt::BevelJoin)},
  478. {rmg::EConnectionType::WIDE, QPen(Qt::black, 5, Qt::DotLine, Qt::SquareCap, Qt::BevelJoin)},
  479. {rmg::EConnectionType::FORCE_PORTAL, QPen(Qt::black, 5, Qt::DashDotDotLine, Qt::SquareCap, Qt::BevelJoin)}
  480. };
  481. lines[i]->setPen(pens[connection.connectionType]);
  482. }
  483. }
  484. void TemplateEditor::saveContent()
  485. {
  486. templates[selectedTemplate]->name = ui->lineEditName->text().toStdString();
  487. templates[selectedTemplate]->description = ui->lineEditDescription->text().toStdString();
  488. templates[selectedTemplate]->minSize = int3(ui->spinBoxMinSizeX->value(), ui->spinBoxMinSizeY->value(), ui->spinBoxMinSizeZ->value());
  489. templates[selectedTemplate]->maxSize = int3(ui->spinBoxMaxSizeX->value(), ui->spinBoxMaxSizeY->value(), ui->spinBoxMaxSizeZ->value());
  490. templates[selectedTemplate]->allowedWaterContent.clear();
  491. if(ui->checkBoxAllowedWaterContentNone->isChecked())
  492. templates[selectedTemplate]->allowedWaterContent.insert(EWaterContent::EWaterContent::NONE);
  493. if(ui->checkBoxAllowedWaterContentNormal->isChecked())
  494. templates[selectedTemplate]->allowedWaterContent.insert(EWaterContent::EWaterContent::NORMAL);
  495. if(ui->checkBoxAllowedWaterContentIslands->isChecked())
  496. templates[selectedTemplate]->allowedWaterContent.insert(EWaterContent::EWaterContent::ISLANDS);
  497. changed();
  498. }
  499. bool TemplateEditor::getAnswerAboutUnsavedChanges()
  500. {
  501. if(unsaved)
  502. {
  503. auto sure = QMessageBox::question(this, tr("Confirmation"), tr("Unsaved changes will be lost, are you sure?"));
  504. if(sure == QMessageBox::No)
  505. {
  506. return false;
  507. }
  508. }
  509. return true;
  510. }
  511. void TemplateEditor::setTitle()
  512. {
  513. QFileInfo fileInfo(filename);
  514. QString title = QString("%1%2 - %3 (%4)").arg(fileInfo.fileName(), unsaved ? "*" : "", tr("VCMI Template Editor"), GameConstants::VCMI_VERSION.c_str());
  515. setWindowTitle(title);
  516. }
  517. void TemplateEditor::changed()
  518. {
  519. unsaved = true;
  520. setTitle();
  521. }
  522. void TemplateEditor::saveTemplate()
  523. {
  524. saveContent();
  525. Helper::saveTemplate(templates, filename);
  526. unsaved = false;
  527. }
  528. void TemplateEditor::showTemplateEditor()
  529. {
  530. auto * dialog = new TemplateEditor();
  531. dialog->setAttribute(Qt::WA_DeleteOnClose);
  532. }
  533. void TemplateEditor::on_actionOpen_triggered()
  534. {
  535. if(!getAnswerAboutUnsavedChanges())
  536. return;
  537. auto filenameSelect = QFileDialog::getOpenFileName(this, tr("Open template"),
  538. QString::fromStdString(VCMIDirs::get().userDataPath().make_preferred().string()),
  539. tr("VCMI templates(*.json)"));
  540. if(filenameSelect.isEmpty())
  541. return;
  542. templates = Helper::openTemplateInternal(filenameSelect);
  543. initContent();
  544. loadContent();
  545. ui->templateView->autoFit();
  546. }
  547. void TemplateEditor::on_actionSave_as_triggered()
  548. {
  549. auto filenameSelect = QFileDialog::getSaveFileName(this, tr("Save template"), "", tr("VCMI templates (*.json)"));
  550. if(filenameSelect.isNull())
  551. return;
  552. QFileInfo fileInfo(filenameSelect);
  553. if(fileInfo.suffix().toLower() != "json")
  554. filenameSelect += ".json";
  555. filename = filenameSelect;
  556. saveTemplate();
  557. setTitle();
  558. }
  559. void TemplateEditor::on_actionNew_triggered()
  560. {
  561. if(!getAnswerAboutUnsavedChanges())
  562. return;
  563. templates = std::map<std::string, std::shared_ptr<CRmgTemplate>>();
  564. templates["TemplateEditor"] = std::make_shared<CRmgTemplate>();
  565. changed();
  566. initContent();
  567. loadContent();
  568. ui->templateView->autoFit();
  569. }
  570. void TemplateEditor::on_actionSave_triggered()
  571. {
  572. if(filename.isNull())
  573. on_actionSave_as_triggered();
  574. else
  575. saveTemplate();
  576. setTitle();
  577. }
  578. void TemplateEditor::on_actionAutoPosition_triggered()
  579. {
  580. if(!templates.size())
  581. return;
  582. saveContent();
  583. loadContent(true);
  584. ui->templateView->autoFit();
  585. }
  586. void TemplateEditor::on_actionZoom_in_triggered()
  587. {
  588. ui->templateView->changeZoomLevel(true);
  589. }
  590. void TemplateEditor::on_actionZoom_out_triggered()
  591. {
  592. ui->templateView->changeZoomLevel(false);
  593. }
  594. void TemplateEditor::on_actionZoom_auto_triggered()
  595. {
  596. ui->templateView->autoFit();
  597. }
  598. void TemplateEditor::on_actionZoom_reset_triggered()
  599. {
  600. ui->templateView->setZoomLevel(0);
  601. }
  602. void TemplateEditor::on_actionAddZone_triggered()
  603. {
  604. if(!templates.size())
  605. return;
  606. for(int i = 1; i < 999; i++)
  607. {
  608. if(!templates[selectedTemplate]->zones.count(i))
  609. {
  610. templates[selectedTemplate]->zones[i] = std::make_shared<rmg::ZoneOptions>();
  611. break;
  612. }
  613. }
  614. loadContent();
  615. }
  616. void TemplateEditor::on_actionRemoveZone_triggered()
  617. {
  618. templates[selectedTemplate]->zones.erase(selectedZone);
  619. selectedZone = -1;
  620. loadContent();
  621. }
  622. void TemplateEditor::on_comboBoxTemplateSelection_activated(int index)
  623. {
  624. auto value = ui->comboBoxTemplateSelection->currentText().toStdString();
  625. if(value.empty())
  626. return;
  627. saveContent();
  628. selectedTemplate = value;
  629. loadContent();
  630. }
  631. void TemplateEditor::closeEvent(QCloseEvent *event)
  632. {
  633. if(getAnswerAboutUnsavedChanges())
  634. QWidget::closeEvent(event);
  635. else
  636. event->ignore();
  637. }
  638. void TemplateEditor::on_pushButtonAddSubTemplate_clicked()
  639. {
  640. bool ok;
  641. QString text = QInputDialog::getText(this, tr("Enter Name"), tr("Name:"), QLineEdit::Normal, "", &ok);
  642. if (!ok || text.isEmpty())
  643. return;
  644. if(templates.count(text.toStdString()))
  645. {
  646. QMessageBox::critical(this, tr("Already existing!"), tr("At template with this name is already existing."));
  647. return;
  648. }
  649. selectedTemplate = text.toStdString();
  650. templates[selectedTemplate] = std::make_shared<CRmgTemplate>();
  651. ui->comboBoxTemplateSelection->addItem(text);
  652. ui->comboBoxTemplateSelection->setCurrentIndex(ui->comboBoxTemplateSelection->count() - 1);
  653. loadContent();
  654. }
  655. void TemplateEditor::on_pushButtonRemoveSubTemplate_clicked()
  656. {
  657. if(templates.size() < 2)
  658. {
  659. QMessageBox::critical(this, tr("To few templates!"), tr("At least one template should remain after removing."));
  660. return;
  661. }
  662. templates.erase(selectedTemplate);
  663. ui->comboBoxTemplateSelection->removeItem(ui->comboBoxTemplateSelection->currentIndex());
  664. selectedTemplate = ui->comboBoxTemplateSelection->currentText().toStdString();
  665. loadContent();
  666. }
  667. void TemplateEditor::on_pushButtonRenameSubTemplate_clicked()
  668. {
  669. if(ui->comboBoxTemplateSelection->currentText().isEmpty() || !templates.size())
  670. return;
  671. bool ok;
  672. QString text = QInputDialog::getText(this, tr("Enter Name"), tr("Name:"), QLineEdit::Normal, ui->comboBoxTemplateSelection->currentText(), &ok);
  673. if (!ok || text.isEmpty())
  674. return;
  675. ui->comboBoxTemplateSelection->setItemText(ui->comboBoxTemplateSelection->currentIndex(), text);
  676. templates[text.toStdString()] = templates[selectedTemplate];
  677. templates.erase(selectedTemplate);
  678. selectedTemplate = text.toStdString();
  679. }
  680. void TemplateEditor::on_spinBoxZoneVisPosX_valueChanged()
  681. {
  682. if(ui->spinBoxZoneVisPosX->hasFocus())
  683. saveZoneMenuContent();
  684. }
  685. void TemplateEditor::on_spinBoxZoneVisPosY_valueChanged()
  686. {
  687. if(ui->spinBoxZoneVisPosY->hasFocus())
  688. saveZoneMenuContent();
  689. }
  690. void TemplateEditor::on_doubleSpinBoxZoneVisSize_valueChanged()
  691. {
  692. if(ui->doubleSpinBoxZoneVisSize->hasFocus())
  693. saveZoneMenuContent();
  694. }
  695. void TemplateEditor::on_comboBoxZoneType_currentTextChanged(const QString &text)
  696. {
  697. if(!ui->comboBoxZoneType->hasFocus())
  698. return;
  699. ui->comboBoxZoneType->clearFocus();
  700. saveZoneMenuContent();
  701. loadZoneMenuContent();
  702. }
  703. void TemplateEditor::on_comboBoxZoneOwner_currentTextChanged(const QString &text)
  704. {
  705. if(ui->comboBoxZoneOwner->hasFocus())
  706. saveZoneMenuContent();
  707. }
  708. void TemplateEditor::on_spinBoxZoneSize_valueChanged()
  709. {
  710. if(ui->spinBoxZoneSize->hasFocus())
  711. saveZoneMenuContent();
  712. }
  713. void TemplateEditor::on_spinBoxTownCountPlayer_valueChanged()
  714. {
  715. if(ui->spinBoxTownCountPlayer->hasFocus())
  716. saveZoneMenuContent();
  717. }
  718. void TemplateEditor::on_spinBoxCastleCountPlayer_valueChanged()
  719. {
  720. if(ui->spinBoxCastleCountPlayer->hasFocus())
  721. saveZoneMenuContent();
  722. }
  723. void TemplateEditor::on_spinBoxTownDensityPlayer_valueChanged()
  724. {
  725. if(ui->spinBoxTownDensityPlayer->hasFocus())
  726. saveZoneMenuContent();
  727. }
  728. void TemplateEditor::on_spinBoxCastleDensityPlayer_valueChanged()
  729. {
  730. if(ui->spinBoxCastleDensityPlayer->hasFocus())
  731. saveZoneMenuContent();
  732. }
  733. void TemplateEditor::on_spinBoxTownCountNeutral_valueChanged()
  734. {
  735. if(ui->spinBoxTownCountNeutral->hasFocus())
  736. saveZoneMenuContent();
  737. }
  738. void TemplateEditor::on_spinBoxCastleCountNeutral_valueChanged()
  739. {
  740. if(ui->spinBoxCastleCountNeutral->hasFocus())
  741. saveZoneMenuContent();
  742. }
  743. void TemplateEditor::on_spinBoxTownDensityNeutral_valueChanged()
  744. {
  745. if(ui->spinBoxTownDensityNeutral->hasFocus())
  746. saveZoneMenuContent();
  747. }
  748. void TemplateEditor::on_spinBoxCastleDensityNeutral_valueChanged()
  749. {
  750. if(ui->spinBoxCastleDensityNeutral->hasFocus())
  751. saveZoneMenuContent();
  752. }
  753. void TemplateEditor::on_checkBoxMatchTerrainToTown_stateChanged(int state)
  754. {
  755. if(ui->checkBoxMatchTerrainToTown->hasFocus())
  756. saveZoneMenuContent();
  757. }
  758. void TemplateEditor::on_checkBoxTownsAreSameType_stateChanged(int state)
  759. {
  760. if(ui->checkBoxTownsAreSameType->hasFocus())
  761. saveZoneMenuContent();
  762. }
  763. void TemplateEditor::on_comboBoxMonsterStrength_currentTextChanged(const QString &text)
  764. {
  765. if(ui->comboBoxMonsterStrength->hasFocus())
  766. saveZoneMenuContent();
  767. }
  768. void TemplateEditor::on_spinBoxZoneId_valueChanged()
  769. {
  770. if(ui->spinBoxZoneId->hasFocus())
  771. saveZoneMenuContent();
  772. }
  773. void TemplateEditor::on_spinBoxZoneLinkTowns_valueChanged()
  774. {
  775. if(ui->spinBoxZoneLinkTowns->hasFocus())
  776. saveZoneMenuContent();
  777. }
  778. void TemplateEditor::on_spinBoxZoneLinkMines_valueChanged()
  779. {
  780. if(ui->spinBoxZoneLinkMines->hasFocus())
  781. saveZoneMenuContent();
  782. }
  783. void TemplateEditor::on_spinBoxZoneLinkTerrain_valueChanged()
  784. {
  785. if(ui->spinBoxZoneLinkTerrain->hasFocus())
  786. saveZoneMenuContent();
  787. }
  788. void TemplateEditor::on_spinBoxZoneLinkTreasure_valueChanged()
  789. {
  790. if(ui->spinBoxZoneLinkTreasure->hasFocus())
  791. saveZoneMenuContent();
  792. }
  793. void TemplateEditor::on_spinBoxZoneLinkCustomObjects_valueChanged()
  794. {
  795. if(ui->spinBoxZoneLinkCustomObjects->hasFocus())
  796. saveZoneMenuContent();
  797. }
  798. void TemplateEditor::on_checkBoxZoneLinkTowns_stateChanged(int state)
  799. {
  800. if(!ui->checkBoxZoneLinkTowns->hasFocus())
  801. return;
  802. ui->checkBoxZoneLinkTowns->clearFocus();
  803. saveZoneMenuContent();
  804. loadZoneMenuContent();
  805. }
  806. void TemplateEditor::on_checkBoxZoneLinkMines_stateChanged(int state)
  807. {
  808. if(!ui->checkBoxZoneLinkMines->hasFocus())
  809. return;
  810. ui->checkBoxZoneLinkMines->clearFocus();
  811. saveZoneMenuContent();
  812. loadZoneMenuContent();
  813. }
  814. void TemplateEditor::on_checkBoxZoneLinkTerrain_stateChanged(int state)
  815. {
  816. if(!ui->checkBoxZoneLinkTerrain->hasFocus())
  817. return;
  818. ui->checkBoxZoneLinkTerrain->clearFocus();
  819. saveZoneMenuContent();
  820. loadZoneMenuContent();
  821. }
  822. void TemplateEditor::on_checkBoxZoneLinkTreasure_stateChanged(int state)
  823. {
  824. if(!ui->checkBoxZoneLinkTreasure->hasFocus())
  825. return;
  826. ui->checkBoxZoneLinkTreasure->clearFocus();
  827. saveZoneMenuContent();
  828. loadZoneMenuContent();
  829. }
  830. void TemplateEditor::on_checkBoxZoneLinkCustomObjects_stateChanged(int state)
  831. {
  832. if(!ui->checkBoxZoneLinkCustomObjects->hasFocus())
  833. return;
  834. ui->checkBoxZoneLinkCustomObjects->clearFocus();
  835. saveZoneMenuContent();
  836. loadZoneMenuContent();
  837. }
  838. void TemplateEditor::on_checkBoxAllowedWaterContentNone_stateChanged(int state)
  839. {
  840. if(ui->checkBoxAllowedWaterContentNone->hasFocus())
  841. saveZoneMenuContent();
  842. }
  843. void TemplateEditor::on_checkBoxAllowedWaterContentNormal_stateChanged(int state)
  844. {
  845. if(ui->checkBoxAllowedWaterContentNormal->hasFocus())
  846. saveZoneMenuContent();
  847. }
  848. void TemplateEditor::on_checkBoxAllowedWaterContentIslands_stateChanged(int state)
  849. {
  850. if(ui->checkBoxAllowedWaterContentIslands->hasFocus())
  851. saveZoneMenuContent();
  852. }
  853. void TemplateEditor::on_pushButtonConnectionAdd_clicked()
  854. {
  855. auto & connections = templates[selectedTemplate]->connections;
  856. connections.push_back(rmg::ZoneConnection());
  857. loadZoneConnectionMenuContent();
  858. }
  859. void TemplateEditor::on_pushButtonOpenTerrainTypes_clicked()
  860. {
  861. TerrainSelector::showTerrainSelector(templates[selectedTemplate]->getZones().at(selectedZone)->terrainTypes);
  862. }
  863. void TemplateEditor::on_pushButtonOpenBannedTerrainTypes_clicked()
  864. {
  865. TerrainSelector::showTerrainSelector(templates[selectedTemplate]->getZones().at(selectedZone)->bannedTerrains);
  866. }
  867. void TemplateEditor::on_pushButtonAllowedTowns_clicked()
  868. {
  869. FactionSelector::showFactionSelector(templates[selectedTemplate]->getZones().at(selectedZone)->townTypes);
  870. }
  871. void TemplateEditor::on_pushButtonBannedTowns_clicked()
  872. {
  873. FactionSelector::showFactionSelector(templates[selectedTemplate]->getZones().at(selectedZone)->bannedTownTypes);
  874. }
  875. void TemplateEditor::on_pushButtonTownHints_clicked()
  876. {
  877. //TODO: Implement dialog
  878. QMessageBox::critical(this, tr("Error"), tr("Not implemented yet!"));
  879. }
  880. void TemplateEditor::on_pushButtonAllowedMonsters_clicked()
  881. {
  882. FactionSelector::showFactionSelector(templates[selectedTemplate]->getZones().at(selectedZone)->monsterTypes);
  883. }
  884. void TemplateEditor::on_pushButtonBannedMonsters_clicked()
  885. {
  886. FactionSelector::showFactionSelector(templates[selectedTemplate]->getZones().at(selectedZone)->bannedMonsters);
  887. }
  888. void TemplateEditor::on_pushButtonTreasure_clicked()
  889. {
  890. //TODO: Implement dialog
  891. QMessageBox::critical(this, tr("Error"), tr("Not implemented yet!"));
  892. }
  893. void TemplateEditor::on_pushButtonMines_clicked()
  894. {
  895. MineSelector::showMineSelector(templates[selectedTemplate]->getZones().at(selectedZone)->mines);
  896. updateZoneCards(selectedZone);
  897. }
  898. void TemplateEditor::on_pushButtonCustomObjects_clicked()
  899. {
  900. //TODO: Implement dialog
  901. QMessageBox::critical(this, tr("Error"), tr("Not implemented yet!"));
  902. }