18 #define GETTEXT_DOMAIN "wesnoth-lib"
47 #include <boost/algorithm/string/replace.hpp>
48 #include <boost/filesystem.hpp>
73 if (x.first.find(
"INTERNAL") == std::string::npos) {
78 connect_signal<event::SDL_KEY_DOWN>(std::bind(
86 button& quit = find_widget<button>(
"exit");
95 menu_button& alignments = find_widget<menu_button>(
"alignment_list");
99 const std::string& icon_path =
"icons/alignments/alignment_" + std::string(align) +
"_30.png";
100 align_list_.emplace_back(
"label",
t_string(
static_cast<std::string
>(align),
"wesnoth"),
"icon", icon_path);
104 menu_button& races = find_widget<menu_button>(
"race_list");
106 const std::string& race_name =
i.second.id();
107 race_list_.emplace_back(
"label", race_name,
"icon",
i.second.get_icon_path_stem() +
"_30.png");
114 button&
load = find_widget<button>(
"load_unit_type");
126 find_widget<button>(
"browse_unit_image"),
129 find_widget<button>(
"preview_unit_image"),
132 find_widget<button>(
"browse_portrait_image"),
135 find_widget<button>(
"preview_portrait_image"),
139 find_widget<text_box>(
"name_box"),
142 find_widget<text_box>(
"id_box"),
150 menu_button& movetypes = find_widget<menu_button>(
"movetype_list");
159 menu_button& defenses = find_widget<menu_button>(
"defense_list");
168 menu_button& movement_costs = find_widget<menu_button>(
"movement_costs_list");
176 menu_button& resistances = find_widget<menu_button>(
"resistances_list");
183 resistances_list_.emplace_back(
"label", key,
"icon",
"icons/profiles/" + key +
".png");
191 menu_button& usage_types = find_widget<menu_button>(
"usage_list");
199 multimenu_button& abilities = find_widget<multimenu_button>(
"abilities_list");
203 find_widget<button>(
"browse_small_profile_image"),
206 find_widget<button>(
"preview_small_profile_image"),
209 find_widget<button>(
"load_movetype"),
212 find_widget<slider>(
"resistances_slider"),
215 find_widget<menu_button>(
"resistances_list"),
218 find_widget<toggle_button>(
"resistances_checkbox"),
222 find_widget<slider>(
"defense_slider"),
225 find_widget<menu_button>(
"defense_list"),
228 find_widget<toggle_button>(
"defense_checkbox"),
232 find_widget<slider>(
"movement_costs_slider"),
235 find_widget<menu_button>(
"movement_costs_list"),
238 find_widget<toggle_button>(
"movement_costs_checkbox"),
257 multimenu_button& specials = find_widget<multimenu_button>(
"weapon_specials_list");
260 combobox& attack_types = find_widget<combobox>(
"attack_type_list");
267 find_widget<button>(
"browse_attack_image"),
270 find_widget<button>(
"preview_attack_image"),
273 find_widget<menu_button>(
"atk_list"),
276 find_widget<button>(
"atk_new"),
279 find_widget<button>(
"atk_delete"),
282 find_widget<button>(
"atk_next"),
285 find_widget<button>(
"atk_prev"),
316 std::string dn = dlg.
path();
318 =
_(
"This file is outside Wesnoth’s data dirs. Do you wish to copy it into your add-on?");
320 if(id_stem ==
"unit_image") {
328 }
else if((id_stem ==
"portrait_image")||(id_stem ==
"small_profile_image")) {
336 }
else if(id_stem ==
"attack_image") {
346 find_widget<text_box>(
"path_"+id_stem).set_value(dn);
355 if (!type_select->show() && !type_select->is_selected()) {
359 const auto&
type = all_type_list[type_select->get_selected_index()];
364 find_widget<text_box>(
"id_box").set_value(
type->id());
365 find_widget<text_box>(
"name_box").set_value(
type->type_name().base_str());
366 find_widget<spinner>(
"level_box").set_value(
type->level());
367 find_widget<slider>(
"cost_slider").set_value(
type->cost());
368 find_widget<text_box>(
"adv_box").set_value(
utils::join(
type->advances_to()));
369 find_widget<slider>(
"hp_slider").set_value(
type->hitpoints());
370 find_widget<slider>(
"xp_slider").set_value(
type->experience_needed());
371 find_widget<slider>(
"move_slider").set_value(
type->movement());
372 find_widget<scroll_text>(
"desc_box").set_value(
type->unit_description().base_str());
373 find_widget<text_box>(
"path_unit_image").set_value(
type->image());
374 find_widget<text_box>(
"path_portrait_image").set_value(
type->big_profile());
376 for (
const auto& gender :
type->genders())
378 if (gender == unit_race::GENDER::MALE) {
379 find_widget<toggle_button>(
"gender_male").set_value(
true);
382 if (gender == unit_race::GENDER::FEMALE) {
383 find_widget<toggle_button>(
"gender_female").set_value(
true);
388 find_widget<menu_button>(
"race_list"),
393 find_widget<menu_button>(
"alignment_list"),
400 find_widget<text_box>(
"path_small_profile_image").set_value(
type->small_profile());
403 find_widget<menu_button>(
"movetype_list"),
405 type->movement_type_id());
408 type->movement_type().write(cfg,
false);
415 if (!
type->get_cfg().has_child(
"resistance")) {
419 for (
const auto& [key,
_] :
type->get_cfg().mandatory_child(
"resistance").attribute_range()) {
427 if (
type->get_cfg().has_child(
"defense")) {
428 for (
const auto& [key,
_] :
type->get_cfg().mandatory_child(
"defense").attribute_range()) {
435 if (
type->get_cfg().has_child(
"movement_costs")) {
436 for (
const auto& [key,
_] :
type->get_cfg().mandatory_child(
"movement_costs").attribute_range()) {
449 find_widget<menu_button>(
"usage_list"),
457 for(
const auto& atk :
type->attacks())
461 attack[
"name"] = atk.id();
462 attack[
"description"] = atk.name().base_str();
463 attack[
"icon"] = atk.icon();
464 attack[
"range"] = atk.range();
465 attack[
"damage"] = atk.damage();
466 attack[
"number"] = atk.num_attacks();
467 attack[
"type"] = atk.type();
468 attacks_.push_back(std::make_pair(enabled, attack));
471 if (!
type->attacks().empty()) {
490 std::string current_textdomain =
"wesnoth-"+
addon_id_;
515 utype[
"gender"] =
"male,female";
517 utype[
"gender"] =
"male";
521 utype[
"gender"] =
"female";
566 if (abilities_states.any()) {
570 if (
i >= abilities_states.size()) {
574 if (abilities_states[
i] ==
true) {
586 find_widget<slider>(
"resistances_slider")
588 100 -
resistances_[find_widget<menu_button>(
"resistances_list").get_value_string()].to_int());
590 find_widget<slider>(
"resistances_slider")
591 .set_active(
res_toggles_[find_widget<menu_button>(
"resistances_list").get_value()]);
593 find_widget<toggle_button>(
"resistances_checkbox")
594 .set_value(
res_toggles_[find_widget<menu_button>(
"resistances_list").get_value()]);
598 resistances_[find_widget<menu_button>(
"resistances_list").get_value_string()]
599 = 100 - find_widget<slider>(
"resistances_slider").get_value();
603 bool toggle = find_widget<toggle_button>(
"resistances_checkbox").get_value();
604 res_toggles_[find_widget<menu_button>(
"resistances_list").get_value()] = toggle;
605 find_widget<slider>(
"resistances_slider").set_active(toggle);
609 find_widget<slider>(
"defense_slider")
611 100 -
defenses_[find_widget<menu_button>(
"defense_list").get_value_string()].to_int());
613 find_widget<slider>(
"defense_slider")
614 .set_active(
def_toggles_[find_widget<menu_button>(
"defense_list").get_value()]);
616 find_widget<toggle_button>(
"defense_checkbox")
617 .set_value(
def_toggles_[find_widget<menu_button>(
"defense_list").get_value()]);
621 defenses_[find_widget<menu_button>(
"defense_list").get_value_string()]
622 = 100 - find_widget<slider>(
"defense_slider").get_value();
626 bool toggle = find_widget<toggle_button>(
"defense_checkbox").get_value();
627 def_toggles_[find_widget<menu_button>(
"defense_list").get_value()] = toggle;
628 find_widget<slider>(
"defense_slider").set_active(toggle);
632 find_widget<slider>(
"movement_costs_slider")
634 movement_[find_widget<menu_button>(
"movement_costs_list").get_value_string()].to_int());
636 find_widget<slider>(
"movement_costs_slider")
637 .set_active(
move_toggles_[find_widget<menu_button>(
"movement_costs_list").get_value()]);
639 find_widget<toggle_button>(
"movement_costs_checkbox")
640 .set_value(
move_toggles_[find_widget<menu_button>(
"movement_costs_list").get_value()]);
644 movement_[find_widget<menu_button>(
"movement_costs_list").get_value_string()]
645 = find_widget<slider>(
"movement_costs_slider").get_value();
649 bool toggle = find_widget<toggle_button>(
"movement_costs_checkbox").get_value();
650 move_toggles_[find_widget<menu_button>(
"movement_costs_list").get_value()] = toggle;
651 find_widget<slider>(
"movement_costs_slider").set_active(toggle);
656 std::string current_textdomain =
"wesnoth-"+
addon_id_;
669 attack[
"name"] = find_widget<text_box>(
"atk_id_box").get_value();
670 attack[
"description"] =
t_string(find_widget<text_box>(
"atk_name_box").get_value(), current_textdomain);
671 attack[
"icon"] = find_widget<text_box>(
"path_attack_image").get_value();
672 attack[
"type"] = find_widget<combobox>(
"attack_type_list").get_value();
673 attack[
"damage"] = find_widget<slider>(
"dmg_box").get_value();
674 attack[
"number"] = find_widget<slider>(
"dmg_num_box").get_value();
675 attack[
"range"] = find_widget<combobox>(
"range_list").get_value();
694 find_widget<text_box>(
"atk_id_box").set_value(attack[
"name"]);
695 find_widget<text_box>(
"atk_name_box").set_value(attack[
"description"]);
696 find_widget<text_box>(
"path_attack_image").set_value(attack[
"icon"]);
698 find_widget<slider>(
"dmg_box").set_value(attack[
"damage"].to_int());
699 find_widget<slider>(
"dmg_num_box").set_value(attack[
"number"].to_int());
700 find_widget<combobox>(
"range_list").set_value(attack[
"range"]);
705 find_widget<multimenu_button>(
"weapon_specials_list")
721 std::vector<config> atk_name_list;
722 for(
const auto& atk_data :
attacks_) {
723 atk_name_list.emplace_back(
"label", atk_data.second[
"name"]);
725 menu_button& atk_list = find_widget<menu_button>(
"atk_list");
732 find_widget<label>(
"atk_number").set_label(new_index_str);
739 std::string current_textdomain =
"wesnoth-"+
addon_id_;
747 attack[
"name"] = find_widget<text_box>(
"atk_id_box").get_value();
748 attack[
"description"] =
t_string(find_widget<text_box>(
"atk_name_box").get_value(), current_textdomain);
749 attack[
"icon"] = find_widget<text_box>(
"path_attack_image").get_value();
750 attack[
"type"] = find_widget<combobox>(
"attack_type_list").get_value();
751 attack[
"damage"] = find_widget<slider>(
"dmg_box").get_value();
752 attack[
"number"] = find_widget<slider>(
"dmg_num_box").get_value();
753 attack[
"range"] = find_widget<combobox>(
"range_list").get_value();
759 , std::make_pair(find_widget<multimenu_button>(
"weapon_specials_list").get_toggle_states(), attack));
779 find_widget<button>(
"atk_delete").set_active(
false);
831 .mandatory_child(
"units")
832 .child_range(
"movetype")) {
833 if (
movetype[
"name"] == find_widget<menu_button>(
"movetype_list").get_value_string()) {
855 out <<
"{" << macro_name <<
"}\n";
865 std::stringstream wml_stream;
868 std::string current_textdomain =
"wesnoth-"+
addon_id_;
871 <<
"#textdomain " << current_textdomain <<
"\n"
873 <<
"# This file was generated using the scenario editor.\n"
903 for (
const auto& [key, value] : atk.second.attribute_range()) {
904 if (!value.empty()) {
909 if(atk.first.any()) {
978 find_widget<scroll_text>(
"wml_view").set_label(
generated_wml);
982 std::string rel_path = find_widget<text_box>(
"path_"+id_stem).get_value();
985 if (rel_path.find(
"~") != std::string::npos) {
986 rel_path = rel_path.substr(0, rel_path.find(
"~"));
989 int scale_size = 200;
990 if (rel_path.size() > 0) {
992 float aspect_ratio =
static_cast<float>(img_size.x)/img_size.y;
993 if(img_size.x > scale_size) {
994 rel_path.append(
"~SCALE(" + std::to_string(scale_size) +
"," + std::to_string(scale_size*aspect_ratio) +
")");
995 }
else if (img_size.y > scale_size) {
996 rel_path.append(
"~SCALE(" + std::to_string(scale_size/aspect_ratio) +
"," + std::to_string(scale_size) +
")");
1000 if (id_stem ==
"portrait_image") {
1002 find_widget<image>(
"unit_image").set_label(rel_path);
1004 find_widget<image>(id_stem).set_label(rel_path);
1013 if (!(std::isalnum(
c) ||
c ==
'_' ||
c ==
' ')) {
1022 grid*
grid = find_widget<tab_container>(
"tabs").get_tab_grid(0);
1027 find_widget<button>(
"ok").set_active(!
id.empty() && !name.empty() &&
check_id(
id));
1034 =
_(
"Unsaved changes will be lost. Do you want to leave?");
1045 boost::algorithm::replace_all(
unit_name,
" ",
"_");
1061 const SDL_Keycode key,
1062 SDL_Keymod modifier)
1066 const SDL_Keycode modifier_key = KMOD_GUI;
1070 const SDL_Keycode modifier_key = KMOD_CTRL;
1076 if (modifier & modifier_key) {
Class for writing a config out to a file in pieces.
void close_child(const std::string &key)
void open_child(const std::string &key)
A config object defines a single node in a WML file, with access to child nodes.
std::size_t attribute_count() const
Count the number of non-blank attributes.
config & mandatory_child(config_key_type key, int n=0)
Returns the nth child with the given key, or throws an error if there is none.
const_attr_itors attribute_range() const
config & add_child(config_key_type key)
A class grating read only view to a vector of config objects, viewed as one config with all children ...
const config & mandatory_child(config_key_type key) const
void set_values(const std::vector<::config > &values, unsigned selected=0)
Dialog that allows user to create custom unit types.
unsigned int selected_attack_
0 means there are no attacks.
void write()
Write the cfg file.
void quit_confirmation()
Quit confirmation.
std::vector< config > defense_list_
boost::dynamic_bitset res_toggles_
Used to control checkboxes for various resistances, defences, etc.
const game_config_view & game_config_
std::vector< std::string > sel_abilities_
Need this because can't store macros in config.
std::vector< std::pair< boost::dynamic_bitset<>, config > > attacks_
std::vector< config > abilities_list_
void store_attack()
Callbacks for attack page.
void set_selected_from_string(menu_button &list, std::vector< config > values, std::string item)
std::vector< config > movetype_list_
std::vector< config > race_list_
void update_defenses()
Callbacks for defense list.
std::string generated_wml
Generated WML.
void save_unit_type()
Save Unit Type data to cfg.
std::vector< config > align_list_
void button_state_change()
Callback to enable/disable OK button if ID/Name is invalid.
void signal_handler_sdl_key_down(const event::ui_event, bool &handled, const SDL_Keycode key, SDL_Keymod modifier)
editor_edit_unit(const game_config_view &game_config, const std::string &addon_id)
void select_file(const std::string &default_dir, const std::string &id_stem)
Callback for file select button.
preproc_map specials_map_
boost::dynamic_bitset move_toggles_
void update_movement_costs()
Callbacks for movement list.
std::vector< config > resistances_list_
preproc_map abilities_map_
void enable_defense_slider()
void on_page_select()
Callback when an tab item in the "page" listbox is selected.
void update_resistances()
Callback for resistance list.
bool check_id(const std::string &id)
Utility method to check if ID contains any invalid characters.
const std::string & addon_id_
void store_movement_costs()
void enable_movement_slider()
std::vector< config > specials_list_
void enable_resistances_slider()
boost::dynamic_bitset def_toggles_
void load_unit_type()
Load Unit Type data from cfg.
virtual void pre_show() override
Actions to be taken before showing the window.
void update_wml_view()
Update wml preview.
void write_macro(std::ostream &out, unsigned level, const std::string ¯o_name)
Write macro to a stream at specified tab level.
std::vector< config > usage_type_list_
void load_movetype()
Callback for loading movetype data in UI.
void update_image(const std::string &id_stem)
Callback for image update.
file_dialog & set_ok_label(const std::string &value)
Sets the OK button label.
file_dialog & set_path(const std::string &value)
Sets the initial file selection.
file_dialog & set_title(const std::string &value)
Sets the current dialog title text.
file_dialog & set_read_only(bool value)
Whether to provide user interface elements for manipulating existing objects.
std::string path() const
Gets the current file selection.
Main class to show messages to the user.
@ yes_no_buttons
Shows a yes and no button.
Abstract base class for all modal dialogs.
bool show(const unsigned auto_close_time=0)
Shows the window.
static std::unique_ptr< units_dialog > build_create_dialog(const std::vector< const unit_type * > &types_list)
A container widget that shows one of its pages of widgets depending on which tab the user clicked.
grid * get_tab_grid(unsigned i)
void select_tab(unsigned index)
unsigned get_active_tab_index()
std::string get_value() const
A widget that allows the user to input text in single line.
void set_retval(const int retval, const bool close_window=true)
Sets there return value of the window.
void invalidate_layout()
Updates the size of the window.
static const std::string & type()
Static type getter that does not rely on the widget being constructed.
Generic locator abstracting the location of an image.
The basic "size" of the unit - flying, small land, large land, etc.
const movement_type_map & movement_types() const
const race_map & races() const
const std::vector< const unit_type * > types_list() const
Declarations for File-IO.
static std::string _(const char *str)
bool to_asset_path(std::string &path, const std::string &addon_id, const std::string &asset_type)
Helper function to convert absolute path to wesnoth relative path.
const std::string wml_extension
void copy_file(const std::string &src, const std::string &dest)
Read a file and then writes it back out.
void write_file(const std::string &fname, const std::string &data, std::ios_base::openmode mode)
Throws io_exception if an error occurs.
std::string get_current_editor_dir(const std::string &addon_id)
Game configuration data as global variables.
REGISTER_DIALOG(editor_edit_unit)
ui_event
The event sent to the dispatcher.
void connect_signal_notify_modified(dispatcher &dispatcher, const signal_notification &signal)
Connects a signal handler for getting a notification upon modification.
void connect_signal_mouse_left_click(dispatcher &dispatcher, const signal &signal)
Connects a signal handler for a left mouse button click.
std::vector< game_tip > load(const config &cfg)
Loads the tips from a config.
void show_transient_message(const std::string &title, const std::string &message, const std::string &image, const bool message_use_markup, const bool title_use_markup)
Shows a transient message to the user.
void show_message(const std::string &title, const std::string &msg, const std::string &button_caption, const bool auto_close, const bool message_use_markup, const bool title_use_markup)
Shows a message to the user.
@ OK
Dialog was closed with the OK button.
@ CANCEL
Dialog was closed with the CANCEL button.
point get_size(const locator &i_locator, bool skip_cache)
Returns the width and height of an image.
std::string join(const T &v, const std::string &s=",")
Generates a new string joining container items in a list.
filesystem::scoped_istream preprocess_file(const std::string &fname, preproc_map *defines)
Function to use the WML preprocessor on a file.
static config unit_name(const unit *u)
void read(config &cfg, std::istream &in, abstract_validator *validator)
void write_key_val(std::ostream &out, const std::string &key, const config::attribute_value &value, unsigned level, std::string &textdomain)
An exception object used when an IO error occurs.
static std::string get_string(enum_type key)
Converts a enum to its string equivalent.
unit_type_data unit_types