1use crate::block::Block;
2use crate::ck3::modif::ModifKinds;
3use crate::context::ScopeContext;
4use crate::db::{Db, DbKind};
5use crate::desc::validate_desc;
6use crate::everything::Everything;
7use crate::game::GameFlags;
8use crate::item::{Item, ItemLoader};
9use crate::modif::validate_modifs;
10use crate::scopes::Scopes;
11use crate::token::Token;
12use crate::tooltipped::Tooltipped;
13use crate::validate::validate_possibly_named_color;
14use crate::validator::Validator;
15
16#[derive(Clone, Debug)]
17pub struct SubjectContract {}
18#[derive(Clone, Debug)]
19pub struct SubjectContractGroup {}
20
21inventory::submit! {
22 ItemLoader::Normal(GameFlags::Ck3, Item::SubjectContract, SubjectContract::add)
23}
24inventory::submit! {
25 ItemLoader::Normal(GameFlags::Ck3, Item::SubjectContractGroup, SubjectContractGroup::add)
26}
27
28impl SubjectContract {
29 pub fn add(db: &mut Db, key: Token, block: Block) {
30 db.add(Item::SubjectContract, key, block, Box::new(Self {}));
31 }
32}
33
34impl SubjectContractGroup {
35 pub fn add(db: &mut Db, key: Token, block: Block) {
36 db.add(Item::SubjectContractGroup, key, block, Box::new(Self {}));
37 }
38}
39
40impl DbKind for SubjectContract {
41 fn add_subitems(&self, _key: &Token, block: &Block, db: &mut Db) {
42 if let Some(block) = block.get_field_block("obligation_levels") {
43 for (key, block) in block.iter_definitions() {
44 for token in block.get_field_values("flag") {
45 db.add_flag(Item::SubjectContractFlag, token.clone());
46 }
47 if !key.is("default") {
48 db.add_flag(Item::SubjectContractObligationLevel, key.clone());
49 }
50 }
51 }
52 }
53
54 fn validate(&self, key: &Token, block: &Block, data: &Everything) {
55 data.verify_exists(Item::Localization, key);
56
57 let mut vd = Validator::new(block, data);
58 let mut sc = ScopeContext::new(Scopes::None, key);
59 sc.define_name("liege", Scopes::Character, key);
60 sc.define_name("subject", Scopes::Character, key);
61 sc.define_name("vassal", Scopes::Character, key);
62 sc.define_name("tax_slot", Scopes::TaxSlot, key);
63 sc.define_name("tax_collector", Scopes::Character, key);
64
65 vd.field_bool("uses_opinion_of_liege");
66 if let Some(token) = block.get_field_value("uses_opinion_of_liege") {
67 if token.is("yes") {
68 sc.define_name("opinion_of_liege", Scopes::Value, token);
69 }
70 }
71
72 vd.field_trigger("is_shown", Tooltipped::No, &mut sc);
73 vd.field_choice("display_mode", &["tree", "radiobutton", "checkbox", "hidden"]);
74 vd.field_bool("defaults_to_highest_valid_level");
75 vd.field_trigger("can_be_changed", Tooltipped::Yes, &mut sc);
76 vd.field_item("icon", Item::TextIcon);
77
78 vd.field_validated_block("obligation_levels", |block, data| {
79 let mut vd = Validator::new(block, data);
80 vd.unknown_block_fields(|key, block| {
81 if !key.is("default") {
82 data.verify_exists(Item::Localization, key);
83 let loca = format!("{key}_short");
84 data.verify_exists_implied(Item::Localization, &loca, key);
85 data.localization.suggest(&format!("{key}_desc"), key);
86 }
87
88 let mut vd = Validator::new(block, data);
89 vd.field_bool("default");
90 vd.field_item("parent", Item::SubjectContractObligationLevel);
91 vd.field_list_numeric_exactly("position", 2);
92 vd.field_item("icon", Item::File);
93 vd.field_script_value("levies", &mut sc);
94 vd.field_script_value("tax", &mut sc);
95 vd.field_script_value("herd", &mut sc);
96 vd.field_script_value("barter_goods", &mut sc);
97 vd.field_script_value("piety", &mut sc);
98 vd.field_script_value("prestige", &mut sc);
99 vd.field_script_value("min_levies", &mut sc);
100 vd.field_script_value("min_tax", &mut sc);
101 vd.field_script_value("min_herd", &mut sc);
102 vd.field_script_value("min_barter_goods", &mut sc);
103 vd.field_validated_sc("contribution_desc", &mut sc, validate_desc);
104 vd.field_item("tax_contribution_postfix", Item::Localization);
105 vd.field_item("levies_contribution_postfix", Item::Localization);
106 vd.field_item("herd_contribution_postfix", Item::Localization);
107 vd.field_item("unclamped_contribution_label", Item::Localization);
108 vd.field_item("min_contribution_label", Item::Localization);
109 vd.advice_field("vassal_opinion", "replaced with `subject_opinion` in 1.16");
110 vd.field_integer("subject_opinion");
111 vd.multi_field_value("flag");
112 vd.field_list("gui_tags");
113 vd.field_integer("score");
114 vd.field_validated("color", validate_possibly_named_color);
115 vd.field_script_value("ai_liege_desire", &mut sc);
116 vd.advice_field("ai_vassal_desire", "replaced with `ai_subject_desire` in 1.16");
117 vd.field_script_value("ai_subject_desire", &mut sc);
118 vd.field_validated_block("liege_modifier", |block, data| {
119 let vd = Validator::new(block, data);
120 validate_modifs(block, data, ModifKinds::Character, vd);
121 });
122 vd.advice_field("vassal_modifier", "replaced with `subject_modifier` in 1.16");
123 vd.field_validated_block("subject_modifier", |block, data| {
124 let vd = Validator::new(block, data);
125 validate_modifs(block, data, ModifKinds::Character, vd);
126 });
127 vd.field_trigger("is_shown", Tooltipped::No, &mut sc);
128 vd.field_trigger("is_valid", Tooltipped::Yes, &mut sc);
129 vd.field_bool("enable_title_maa");
130 vd.field_bool("enable_character_maa");
131 vd.field_script_value("tax_factor", &mut sc);
132 vd.field_script_value("levies_factor", &mut sc);
133 vd.field_script_value("herd_factor", &mut sc);
134 vd.field_item("appointment_trait_flag", Item::TraitFlag);
135 });
136 });
137 }
138}
139
140impl DbKind for SubjectContractGroup {
141 fn validate(&self, key: &Token, block: &Block, data: &Everything) {
142 data.verify_exists(Item::Localization, key);
143 let loca = format!("{key}_desc");
144 data.verify_exists_implied(Item::Localization, &loca, key);
145
146 let mut vd = Validator::new(block, data);
147
148 vd.field_item("admin_province_contract", Item::SubjectContract);
149 vd.field_list_items("contracts", Item::SubjectContract);
150 vd.field_value("modify_contract_layout");
151 vd.field_bool("is_tributary");
152
153 let mut sc = ScopeContext::new(Scopes::Character, key);
154 sc.define_name("suzerain", Scopes::Character, key);
155
156 if block.get_field_bool("is_tributary").unwrap_or(false) {
157 vd.field_trigger("is_valid_tributary_contract", Tooltipped::Yes, &mut sc);
158 vd.field_trigger("tributary_can_break_free", Tooltipped::Yes, &mut sc);
159 vd.field_item("suzerain_line_type", Item::LineType);
160 vd.field_item("tributary_line_type", Item::LineType);
161 vd.field_bool("should_show_as_suzerain_realm_name");
162 vd.field_bool("should_show_as_suzerain_realm_color");
163 vd.field_bool("tributary_heir_succession");
164 vd.field_bool("suzerain_heir_succession");
165 } else {
166 vd.ban_field("is_valid_tributary_contract", || "is_tributary = yes");
167 vd.ban_field("tributary_can_break_free", || "is_tributary = yes");
168 vd.ban_field("suzerain_line_type", || "is_tributary = yes");
169 vd.ban_field("tributary_line_type", || "is_tributary = yes");
170 vd.ban_field("should_show_as_suzerain_realm_name", || "is_tributary = yes");
171 vd.ban_field("should_show_as_suzerain_realm_color", || "is_tributary = yes");
172 vd.ban_field("tributary_heir_succession", || "is_tributary = yes");
173 vd.ban_field("suzerain_heir_succession", || "is_tributary = yes");
174 }
175 }
176}