use crate::block::Block;
use crate::context::ScopeContext;
use crate::db::{Db, DbKind};
use crate::everything::Everything;
use crate::game::GameFlags;
use crate::item::{Item, ItemLoader};
use crate::modif::validate_modifs;
use crate::scopes::Scopes;
use crate::token::Token;
use crate::tooltipped::Tooltipped;
use crate::validator::{Validator, ValueValidator};
use crate::vic3::modif::ModifKinds;
#[derive(Clone, Debug)]
pub struct DiplomaticAction {}
inventory::submit! {
ItemLoader::Normal(GameFlags::Vic3, Item::DiplomaticAction, DiplomaticAction::add)
}
impl DiplomaticAction {
pub fn add(db: &mut Db, key: Token, block: Block) {
db.add(Item::DiplomaticAction, key, block, Box::new(Self {}));
}
}
impl DbKind for DiplomaticAction {
fn validate(&self, key: &Token, block: &Block, data: &Everything) {
let mut vd = Validator::new(block, data);
let mut sc = ScopeContext::new(Scopes::Country, key);
sc.define_name("target_country", Scopes::Country, key);
sc.define_name("actor", Scopes::Country, key);
data.verify_exists(Item::Localization, key);
let loca = format!("{key}_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
if block.get_field_bool("requires_approval").unwrap_or(true) {
let loca = format!("{key}_action_name");
data.verify_exists_implied(Item::Localization, &loca, key);
if block.get_field_bool("show_in_lens").unwrap_or(true) {
let loca = format!("{key}_proposal_accepted_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_accepted_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_declined_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_declined_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_notification_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_notification_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_third_party_accepted_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_third_party_accepted_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_third_party_declined_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_proposal_third_party_declined_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
}
} else {
let loca = format!("{key}_action_notification_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_action_notification_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
if block.get_field_bool("should_notify_third_parties").unwrap_or(false) {
let loca = format!("{key}_action_notification_third_party_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_action_notification_third_party_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
}
}
if block.has_key("pact") {
let loca = format!("{key}_pact_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_action_propose_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_action_break_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_action_notification_break_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_action_notification_break_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
if block.get_field_bool("should_notify_third_parties").unwrap_or(false) {
let loca = format!("{key}_action_notification_third_party_break_name");
data.verify_exists_implied(Item::Localization, &loca, key);
let loca = format!("{key}_action_notification_third_party_break_desc");
data.verify_exists_implied(Item::Localization, &loca, key);
}
}
vd.field_validated_list("groups", |token, data| {
let mut vd = ValueValidator::new(token, data);
vd.choice(&[
"general",
"subject",
"overlord",
"power_bloc",
"power_bloc_leader",
"power_bloc_member",
]);
});
vd.field_bool("requires_approval");
vd.field_bool("uses_random_approval");
vd.field_bool("show_confirmation_box");
vd.field_bool("is_hostile");
vd.field_bool("should_notify_third_parties"); vd.field_bool("show_effect_in_tooltip"); vd.field_bool("violates_sovereignty"); vd.field_bool("can_use_obligations"); vd.field_bool("can_select"); vd.field_bool("can_select_to_break"); vd.field_bool("show_in_lens");
vd.field_choice(
"state_selection",
&[
"first_required",
"first_optional",
"second_required",
"second_optional",
"both_required",
"both_optional",
"any_required",
],
);
for field in &["first_state_list", "second_state_list"] {
vd.field_choice(
field,
&[
"first_country",
"second_country",
"all",
"first_country_and_subjects",
"second_country_and_subjects",
],
);
}
vd.multi_field_list_items("unlocking_technologies", Item::Technology);
vd.field_trigger("selectable", Tooltipped::No, &mut sc);
vd.field_trigger("potential", Tooltipped::No, &mut sc);
vd.field_trigger("possible", Tooltipped::Yes, &mut sc);
for field in &["first_state_trigger", "second_state_trigger"] {
vd.field_trigger_builder(field, Tooltipped::Yes, |key| {
let mut sc = ScopeContext::new(Scopes::State, key);
sc.define_name("country", Scopes::Country, key);
sc.define_name("target_country", Scopes::Country, key);
sc
});
}
vd.field_effect("accept_effect", Tooltipped::Yes, &mut sc);
vd.field_effect("decline_effect", Tooltipped::Yes, &mut sc);
vd.advice_field("effect", "the docs say effect but the field is called accept_effect");
vd.field_validated_block_sc("pact", &mut sc, validate_pact);
vd.field_validated_block_sc("ai", &mut sc, validate_ai);
vd.field_item("reverse_pact", Item::DiplomaticAction); vd.field_item("transfer_pact", Item::DiplomaticAction);
vd.field_item("confirmation_sound", Item::Sound);
vd.field_item("request_sound", Item::Sound);
vd.field_item("hostile_sound", Item::Sound);
vd.field_item("benign_sound", Item::Sound);
vd.field_item("texture", Item::File);
}
}
fn validate_pact(block: &Block, data: &Everything, sc: &mut ScopeContext) {
let mut vd = Validator::new(block, data);
vd.field_numeric("cost");
vd.field_bool("counts_for_tech_spread");
vd.field_bool("show_in_outliner"); vd.field_bool("can_be_used_in_sway_offers"); vd.field_integer("sway_maneuvers_cost"); vd.field_bool("infamy_affects_maintenance"); vd.field_bool("has_junior_participant"); vd.field_bool("is_two_sided_pact"); vd.field_bool("is_alliance"); vd.field_bool("is_defensive_pact"); vd.field_bool("is_rivalry"); vd.field_bool("is_embargo"); vd.field_bool("is_trade_agreement"); vd.field_bool("is_customs_union"); vd.field_bool("is_humiliation"); vd.field_bool("is_colonization_rights"); vd.field_bool("is_hostile"); vd.field_bool("is_guarantee_independence"); vd.field_bool("exempt_from_service"); vd.field_bool("is_foreign_investment_rights"); vd.field_bool("junior_support_only_against_overlord");
vd.field_item("subject_type", Item::SubjectType); vd.field_choice("maintenance_paid_by", &["first_country", "second_country"]); vd.advice_field("recipient_pays_maintenance", "removed in 1.9.3");
vd.replaced_field(
"recipient_gets_income_transfer",
"renamed to second_country_gets_income_transfer",
);
vd.field_bool("second_country_gets_income_transfer");
vd.replaced_field(
"income_transfer_based_on_recipient",
"renamed to income_transfer_based_on_second_country",
);
vd.field_bool("income_transfer_based_on_second_country");
vd.field_validated_block("income_transfer_to_pops", |block, data| {
let mut vd = Validator::new(block, data);
vd.replaced_field("allow_discriminated", "allow_non_fully_accepted");
vd.field_bool("allow_non_fully_accepted");
for field in &["upper_strata_pops", "middle_strata_pops", "lower_strata_pops"] {
vd.field_script_value(field, sc);
}
});
vd.field_numeric("income_transfer"); vd.field_numeric("max_paying_country_income_to_transfer");
vd.field_numeric("relations_progress_per_day"); vd.field_numeric("relations_improvement_max"); vd.field_numeric("relations_improvement_min");
vd.field_integer("forced_duration");
vd.field_item("propose_string", Item::Localization);
vd.field_item("break_string", Item::Localization);
vd.field_item("ask_to_end_string", Item::Localization);
vd.field_bool("military_access");
vd.field_bool("actor_requires_approval_to_break");
vd.field_bool("target_requires_approval_to_break");
vd.field_bool("is_breaking_hostile");
vd.field_bool("is_target_breaking_hostile");
vd.replaced_field(
"is_about_to_auto_break",
"show_about_to_break_warning in requirement_to_maintain",
);
vd.replaced_field("should_auto_break", "trigger in requirement_to_maintain");
vd.replaced_field("should_invalidate", "trigger in requirement_to_maintain");
vd.field_trigger("actor_can_break", Tooltipped::No, sc);
vd.field_trigger("target_can_break", Tooltipped::No, sc);
vd.multi_field_validated_block("requirement_to_maintain", |block, data| {
let mut vd = Validator::new(block, data);
vd.field_trigger("trigger", Tooltipped::No, sc);
vd.field_trigger("show_about_to_break_warning", Tooltipped::No, sc);
});
vd.replaced_field("break_effect", "manual_break_effect and auto_break_effect");
for field in &[
"daily_effect",
"weekly_effect",
"monthly_effect",
"manual_break_effect",
"auto_break_effect",
] {
vd.field_effect(field, Tooltipped::No, sc);
}
vd.field_validated_block("subject_relation", |block, data| {
let mut vd = Validator::new(block, data);
vd.field_bool("annex_on_country_formation");
});
vd.multi_field_item("auto_support_type", Item::DiplomaticPlay);
for field in &["first_modifier", "second_modifier"] {
vd.field_validated_block(field, |block, data| {
let vd = Validator::new(block, data);
validate_modifs(block, data, ModifKinds::Country, vd);
});
}
for field in &[
"first_foreign_pro_country_lobby_member_modifier",
"first_foreign_anti_country_lobby_member_modifier",
"second_foreign_pro_country_lobby_member_modifier",
"second_foreign_anti_country_lobby_member_modifier",
] {
vd.field_validated_block(field, |block, data| {
let vd = Validator::new(block, data);
validate_modifs(block, data, ModifKinds::InterestGroup, vd);
});
}
vd.field_choice("market_owner", &["first_country", "second_country"]);
}
fn validate_ai(block: &Block, data: &Everything, sc: &mut ScopeContext) {
let mut vd = Validator::new(block, data);
vd.field_script_value_rooted("evaluation_chance", Scopes::Country);
vd.field_trigger_rooted("will_select_as_first_state", Tooltipped::No, Scopes::State);
vd.field_trigger_rooted("will_select_as_second_state", Tooltipped::No, Scopes::State);
vd.field_bool("check_acceptance_for_will_break");
vd.field_bool("check_acceptance_for_will_propose");
vd.field_numeric_range("max_influence_spending_fraction", 0.0..=1.0);
for field in &[
"will_propose_with_states",
"will_propose",
"will_break",
"will_propose_even_if_not_accepted",
] {
vd.field_trigger(field, Tooltipped::No, sc);
}
for field in &[
"accept_score",
"accept_break_score",
"propose_score",
"propose_break_score",
"use_obligation_chance",
"owe_obligation_chance",
"junior_accept_score",
] {
vd.field_script_value(field, sc);
}
vd.advice_field("use_favor_chance", "this field is called use_obligation_chance");
vd.advice_field("owe_favor_chance", "this field is called owe_obligation_chance");
}