From 633a4010a3d39d3c9c51670cdfd548a451927fdc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C5=82awomir=20Lach?= <slawek@lach.art.pl> Date: Mon, 16 May 2022 15:32:56 +0200 Subject: [PATCH] =?UTF-8?q?!OSDN:=20#TICKET:=2045892:=20S=C5=82awomir=20La?= =?UTF-8?q?ch=20<slawek@lach.art.pl>?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Implementing ruledit tab for counters diff --git a/meson.build b/meson.build index e3d59b38e5..c93b701051 100644 --- a/meson.build +++ b/meson.build @@ -3494,6 +3494,7 @@ mocced_ruledit = qt_mod.preprocess( 'tools/ruledit/req_vec_fix.h', 'tools/ruledit/requirers_dlg.h', 'tools/ruledit/ruledit_qt.h', + 'tools/ruledit/tab_counters.h', 'tools/ruledit/tab_enablers.h', 'tools/ruledit/tab_extras.h', 'tools/ruledit/tab_misc.h', @@ -3518,6 +3519,7 @@ executable('freeciv-ruledit', 'tools/ruledit/ruledit.cpp', 'tools/ruledit/ruledit_qt.cpp', 'tools/ruledit/tab_building.cpp', + 'tools/ruledit/tab_counters.cpp', 'tools/ruledit/tab_enablers.cpp', 'tools/ruledit/tab_extras.cpp', 'tools/ruledit/tab_good.cpp', diff --git a/tools/ruledit/Makefile.am b/tools/ruledit/Makefile.am index 44d0288c1e..5ac03ae97e 100644 --- a/tools/ruledit/Makefile.am +++ b/tools/ruledit/Makefile.am @@ -33,6 +33,7 @@ MOC_FILES = \ meta_requirers_dlg.cpp \ meta_ruledit_qt.cpp \ meta_tab_building.cpp \ + meta_tab_counters.cpp \ meta_tab_enablers.cpp \ meta_tab_extras.cpp \ meta_tab_good.cpp \ @@ -55,6 +56,8 @@ freeciv_ruledit_SOURCES = \ effect_edit.h \ tab_building.cpp \ tab_building.h \ + tab_counters.cpp \ + tab_counters.h \ tab_enablers.cpp \ tab_enablers.h \ tab_extras.cpp \ diff --git a/tools/ruledit/ruledit_qt.cpp b/tools/ruledit/ruledit_qt.cpp index 967966dfb7..5b0559dadf 100644 --- a/tools/ruledit/ruledit_qt.cpp +++ b/tools/ruledit/ruledit_qt.cpp @@ -46,6 +46,7 @@ #include "req_vec_fix.h" #include "ruledit.h" #include "tab_building.h" +#include "tab_counters.h" #include "tab_enablers.h" #include "tab_extras.h" #include "tab_good.h" @@ -185,6 +186,8 @@ void ruledit_gui::setup(QWidget *central_in) stack->addTab(terrains, QString::fromUtf8(R__("Terrains"))); multipliers = new tab_multiplier(this); stack->addTab(multipliers, QString::fromUtf8(R__("Multipliers"))); + counter = new tab_counter(this); + stack->addTab(counter, QString::fromUtf8(R__("Counters"))); nation = new tab_nation(this); stack->addTab(nation, QString::fromUtf8(R__("Nations"))); @@ -250,6 +253,7 @@ void ruledit_gui::launch_now() extras->refresh(); multipliers->refresh(); terrains->refresh(); + counter->refresh(); main_layout->setCurrentIndex(1); } else { display_msg(R__("Ruleset loading failed!")); diff --git a/tools/ruledit/ruledit_qt.h b/tools/ruledit/ruledit_qt.h index 464d443083..e65192b316 100644 --- a/tools/ruledit/ruledit_qt.h +++ b/tools/ruledit/ruledit_qt.h @@ -30,6 +30,7 @@ class QStackedLayout; class requirers_dlg; class tab_building; +class tab_counter; class tab_good; class tab_gov; class tab_misc; @@ -130,6 +131,7 @@ signals: QStackedLayout *main_layout; tab_building *bldg; + tab_counter *counter; tab_misc *misc; tab_tech *tech; tab_unit *unit; diff --git a/tools/ruledit/tab_counters.cpp b/tools/ruledit/tab_counters.cpp new file mode 100644 index 0000000000..35a44ad2da --- /dev/null +++ b/tools/ruledit/tab_counters.cpp @@ -0,0 +1,387 @@ +/*********************************************************************** + Freeciv - Copyright (C) 1996 - A Kjeldberg, L Gregersen, P Unold + 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, 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. +***********************************************************************/ + +#ifdef HAVE_CONFIG_H +#include <fc_config.h> +#endif + +// Qt +#include <QCheckBox> +#include <QGridLayout> +#include <QLineEdit> +#include <QPushButton> + +// utility +#include "fcintl.h" +#include "log.h" + +// common +#include "counters.h" +#include "fc_types.h" +#include "game.h" + +// ruledit +#include "ruledit.h" +#include "ruledit_qt.h" +#include "validity.h" + +#include "tab_counters.h" + +/**********************************************************************//** + Setup tab_counter object +**************************************************************************/ +tab_counter::tab_counter(ruledit_gui *ui_in) : QWidget() +{ +#define empty_widget (nullptr) +/* We use const array of pointers and hopes compiler + * will unroll loop + */ +#define widgets_row(...) { \ + column = 0; \ + QWidget * const wid[] = { __VA_ARGS__ }; \ + for (unsigned int i = 0; \ + i < sizeof(wid) / sizeof(wid[0]); i++) {\ + if (wid[i]) { \ + counter_layout->addWidget(wid[i], \ + row, \ + column);\ + }\ + column++;\ + };\ + row++;\ + } + + QVBoxLayout *main_layout = new QVBoxLayout(this); + QGridLayout *counter_layout = new QGridLayout(); + QLabel *label; + QPushButton *add_button; + QPushButton *delete_button; + int row = 0; + int column = 0; + + ui = ui_in; + selected = nullptr; + + counter_list = new QListWidget(this); + + connect(counter_list, SIGNAL(itemSelectionChanged()), this, SLOT(select_counter())); + main_layout->addWidget(counter_list); + + counter_layout->setSizeConstraint(QLayout::SetMaximumSize); + + label = new QLabel(QString::fromUtf8(R__("Rule Name"))); + label->setParent(this); + rname = new QLineEdit(this); + rname->setText(R__("None")); + connect(rname, SIGNAL(returnPressed()), this, SLOT(name_given())); + widgets_row(label, empty_widget, rname); + + label = new QLabel(QString::fromUtf8(R__("Name"))); + label->setParent(this); + + name = new QLineEdit(this); + name->setText(R__("None")); + connect(name, SIGNAL(returnPressed()), this, SLOT(name_given())); + same_name = new QCheckBox(this); + connect(same_name, SIGNAL(toggled(bool)), this, SLOT(same_name_toggle(bool))); + same_name->setChecked(true); + widgets_row(label, same_name, name); + + label = new QLabel(QString::fromUtf8(R__("Default Value")), this); + label->setParent(this); + def = new QSpinBox(this); + def->setMaximum(INT_MAX); + def->setMinimum(INT_MIN); + connect(def, SIGNAL(valueChanged(int)), this, SLOT(default_given(int))); + widgets_row(label, empty_widget, def); + + + label = new QLabel(QString::fromUtf8(R__("Checkpoint")), this); + label->setParent(this); + checkpoint = new QSpinBox(this); + checkpoint->setMaximum(INT_MAX); + checkpoint->setMinimum(INT_MIN); + connect(checkpoint, SIGNAL(valueChanged(int)), this, SLOT(checkpoint_given(int))); + widgets_row(label, empty_widget, checkpoint); + + { + enum counter_behaviour current; + type = new QComboBox(this); + + label = new QLabel(QString::fromUtf8(R__("Behavior"))); + label->setParent(this); + + for (current = counter_behaviour_begin(); current > counter_behaviour_end(); current = counter_behaviour_next(current)) { + QVariant value(current); + + type->addItem(counter_behaviour_name(current), value); + }; + + connect(type, SIGNAL(activated(int)), this, SLOT(counter_behaviour_selected(int))); + widgets_row(label, empty_widget, type); + } + + effects_button = new QPushButton(QString::fromUtf8(R__("Effects")), this); + connect(effects_button, SIGNAL(pressed()), this, SLOT(edit_effects())); + widgets_row(empty_widget, empty_widget, effects_button); + + add_button = new QPushButton(QString::fromUtf8(R__("Add Counter")), this); + connect(add_button, SIGNAL(pressed()), this, SLOT(add_now())); + + delete_button = new QPushButton(QString::fromUtf8(R__("Remove this Counter")), this); + connect(delete_button, SIGNAL(pressed()), this, SLOT(delete_now())); + + widgets_row(empty_widget, empty_widget, add_button, delete_button); + show_experimental(add_button); + show_experimental(delete_button); + + refresh(); + update_counter_info(nullptr); + + main_layout->addLayout(counter_layout); + + setLayout(main_layout); +#undef empty_widget +#undef widgets_row +} + +/**********************************************************************//** + Called when counter behaviour is set by user +**************************************************************************/ +void tab_counter::counter_behaviour_selected(int item) +{ + QVariant item_data; + + if (nullptr == selected) { + + return; + } + item_data = type->currentData(); + selected->type = (enum counter_behaviour) item_data.toInt(); + + update_counter_info(selected); + refresh(); +} + +/**********************************************************************//** + Refresh the information. +**************************************************************************/ +void tab_counter::refresh() +{ + counter_list->clear(); + + counters_re_iterate(pcount) { + QListWidgetItem *item = new QListWidgetItem(counter_rule_name(pcount)); + + counter_list->insertItem(counter_index(pcount), item); + } counters_re_iterate_end; +} + +/**********************************************************************//** + Update info of the counter +**************************************************************************/ +void tab_counter::update_counter_info(struct counter *counter) +{ + selected = counter; + + if (selected != nullptr) { + QString dispn = QString::fromUtf8(untranslated_name(&(counter->name))); + QString rulen = QString::fromUtf8(rule_name_get(&(counter->name))); + + name->setText(dispn); + rname->setText(rulen); + + if (dispn == rulen) { + name->setEnabled(false); + } else { + name->setEnabled(true); + } + + checkpoint->setValue(selected->checkpoint); + def->setValue(selected->def); + } else { + name->setText(R__("None")); + rname->setText(R__("None")); + name->setEnabled(false); + } +} + +/**********************************************************************//** + User selected counter from the list. +**************************************************************************/ +void tab_counter::select_counter() +{ + char *cname; + + QList<QListWidgetItem *> select_list = counter_list->selectedItems(); + + if (!select_list.isEmpty()) { + QByteArray tn_bytes; + + tn_bytes = select_list.at(0)->text().toUtf8(); + + cname = tn_bytes.data(); + update_counter_info(counter_by_rule_name(cname)); + } +} + + +/**********************************************************************//** + User entered checkpoint value +**************************************************************************/ +void tab_counter::checkpoint_given(int val) +{ + if (nullptr != selected) { + + selected->checkpoint = val; + } +} + +/**********************************************************************//** + User entered default value +**************************************************************************/ +void tab_counter::default_given(int val) +{ + if (nullptr != selected) { + + selected->def = val; + } +} + +/**********************************************************************//** + User entered name for counter +**************************************************************************/ +void tab_counter::name_given() +{ + if (selected != nullptr) { + QByteArray name_bytes; + QByteArray rname_bytes; + + city_counters_iterate(pcounter) { + if (pcounter != selected) { + rname_bytes = rname->text().toUtf8(); + if (!strcmp(counter_rule_name(pcounter), rname_bytes.data())) { + ui->display_msg(R__("A counter with that rule name already exists!")); + return; + } + } + } city_counters_iterate_end; + + name_bytes = name->text().toUtf8(); + rname_bytes = rname->text().toUtf8(); + names_set(&(selected->name), 0, + name_bytes.data(), + rname_bytes.data()); + refresh(); + } +} + +/**********************************************************************//** + User requested counter deletion +**************************************************************************/ +void tab_counter::delete_now() +{ + + if (nullptr != selected) { + requirers_dlg *requirers; + + requirers = ui->create_requirers(counter_rule_name(selected)); + if (is_counter_needed(selected, &ruledit_qt_display_requirers, requirers)) { + return; + } + + selected->ruledit_disabled = true; + + refresh(); + update_counter_info(nullptr); + } +} + +/**********************************************************************//** + Initialize new counter for use. +**************************************************************************/ +bool tab_counter::initialize_new_counter(struct counter *counter) +{ + counter->checkpoint = 0; + counter->def = 0; + counter->target = CTGT_CITY; + counter->type = CB_CITY_OWNED_TURNS; + counter->index = counters_get_city_counters_count() - 1; + counter->ruledit_disabled = FALSE; + + name_set(&counter->name, 0, "New counter"); + + return true; +} + +/**********************************************************************//** + User requested new counter +**************************************************************************/ +void tab_counter::add_now() +{ + struct counter *new_counter; + + // Try to reuse freed counter slot + city_counters_iterate(pcount) { + if (pcount->type == COUNTER_BEHAVIOUR_LAST + || pcount->ruledit_disabled) { + if (initialize_new_counter(pcount)) { + update_counter_info(pcount); + refresh(); + } + return; + } + } city_counters_iterate_end; + + // Try to add completely new counter + if (counters_get_city_counters_count() >= MAX_COUNTERS) { + return; + } + + // num_counter_types must be big enough to hold new counter or + // counter_by_number() fails. + game.control.num_counters++; + new_counter = counter_by_id(counters_get_city_counters_count()); + if (initialize_new_counter(new_counter)) { + update_counter_info(new_counter); + attach_city_counter(new_counter); + refresh(); + } +} + +/**********************************************************************//** + Toggled whether rule_name and name should be kept identical +**************************************************************************/ +void tab_counter::same_name_toggle(bool checked) +{ + name->setEnabled(!checked); + if (checked) { + name->setText(rname->text()); + } +} + +/**********************************************************************//** + User wants to edit effects +**************************************************************************/ +void tab_counter::edit_effects() +{ + if (selected != nullptr) { + struct universal uni; + + uni.value.counter = selected; + uni.kind = VUT_COUNTER; + + ui->open_effect_edit(QString::fromUtf8(counter_rule_name(selected)), + &uni, EFMC_NORMAL); + } +} diff --git a/tools/ruledit/tab_counters.h b/tools/ruledit/tab_counters.h new file mode 100644 index 0000000000..8d379d6456 --- /dev/null +++ b/tools/ruledit/tab_counters.h @@ -0,0 +1,75 @@ +/*********************************************************************** + Freeciv - Copyright (C) 1996 - A Kjeldberg, L Gregersen, P Unold + 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, 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. +***********************************************************************/ + +#ifndef FC__TAB_COUNTER_H +#define FC__TAB_COUNTER_H + +#ifdef HAVE_CONFIG_H +#include <fc_config.h> +#endif + +// Qt +#include <QCheckBox> +#include <QComboBox> +#include <QListWidget> +#include <QMenu> +#include <QPushButton> +#include <QSpinBox> +#include <QWidget> + +// common +#include "counters.h" + +class ruledit_gui; + +class tab_counter : public QWidget +{ + Q_OBJECT + + public: + explicit tab_counter(ruledit_gui *ui_in); + void refresh(); + static void techs_to_menu(QMenu *fill_menu); + static QString counter_name(struct counter *padv); + + private: + ruledit_gui *ui; + + QLineEdit *name; + QLineEdit *rname; + QSpinBox *checkpoint; + QSpinBox *def; + QComboBox *type; + QCheckBox *same_name; + QPushButton *effects_button; + + QListWidget *counter_list; + + struct counter *selected; + + private slots: + void update_counter_info(struct counter *counter); + void checkpoint_given(int val); + void default_given(int val); + void name_given(); + void select_counter(); + void add_now(); + void delete_now(); + void same_name_toggle(bool checked); + void edit_effects(); + bool initialize_new_counter(struct counter *padv); + void counter_behaviour_selected(int item); +}; + + +#endif // FC__TAB_COUNTER_H diff --git a/tools/ruleutil/rulesave.c b/tools/ruleutil/rulesave.c index 836d41b462..ee6cb9f81c 100644 --- a/tools/ruleutil/rulesave.c +++ b/tools/ruleutil/rulesave.c @@ -1787,7 +1787,7 @@ static bool save_game_ruleset(const char *filename, const char *name) comment_counters(sfile); sect_idx = 0; - city_counters_iterate(pcounter) { + counters_re_iterate(pcounter) { char path[512]; fc_snprintf(path, sizeof(path), "counter_%d", sect_idx++); @@ -1799,7 +1799,7 @@ static bool save_game_ruleset(const char *filename, const char *name) secfile_insert_str(sfile, counter_behaviour_name(pcounter->type), "%s.type", path); - } city_counters_iterate_end; + } counters_re_iterate_end; locks = FALSE; settings_iterate(SSET_ALL, pset) { diff --git a/translations/ruledit/POTFILES.in b/translations/ruledit/POTFILES.in index 7678c8d1e0..5050794268 100644 --- a/translations/ruledit/POTFILES.in +++ b/translations/ruledit/POTFILES.in @@ -9,6 +9,7 @@ tools/ruledit/requirers_dlg.cpp tools/ruledit/ruledit.cpp tools/ruledit/ruledit_qt.cpp tools/ruledit/tab_building.cpp +tools/ruledit/tab_counters.cpp tools/ruledit/tab_enablers.cpp tools/ruledit/tab_extras.cpp tools/ruledit/tab_good.cpp -- 2.39.1