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_choice("display_mode", &["tree", "list", "radiobutton", "checkbox"]);
73 vd.field_item("icon", Item::TextIcon);
74 vd.field_trigger("is_shown", Tooltipped::No, &mut sc);
75
76 vd.field_validated_block("obligation_levels", |block, data| {
77 let mut vd = Validator::new(block, data);
78 vd.unknown_block_fields(|key, block| {
79 if !key.is("default") {
80 data.verify_exists(Item::Localization, key);
81 let loca = format!("{key}_short");
82 data.verify_exists_implied(Item::Localization, &loca, key);
83 data.localization.suggest(&format!("{key}_desc"), key);
84 }
85
86 let mut vd = Validator::new(block, data);
87 vd.field_bool("default");
88 vd.field_item("parent", Item::SubjectContractObligationLevel);
89 vd.field_list_numeric_exactly("position", 2);
90 vd.field_item("icon", Item::File);
91 vd.field_script_value("levies", &mut sc);
92 vd.field_script_value("tax", &mut sc);
93 vd.field_script_value("herd", &mut sc);
94 vd.field_script_value("prestige", &mut sc);
95 vd.field_script_value("min_levies", &mut sc);
96 vd.field_script_value("min_tax", &mut sc);
97 vd.field_script_value("min_herd", &mut sc);
98 vd.field_validated_sc("contribution_desc", &mut sc, validate_desc);
99 vd.field_item("tax_contribution_postfix", Item::Localization);
100 vd.field_item("levies_contribution_postfix", Item::Localization);
101 vd.field_item("herd_contribution_postfix", Item::Localization);
102 vd.field_item("unclamped_contribution_label", Item::Localization);
103 vd.field_item("min_contribution_label", Item::Localization);
104 vd.advice_field("vassal_opinion", "replaced with `subject_opinion` in 1.16");
105 vd.field_integer("subject_opinion");
106 vd.multi_field_value("flag");
107 vd.field_integer("score");
108 vd.field_validated("color", validate_possibly_named_color);
109 vd.field_script_value("ai_liege_desire", &mut sc);
110 vd.advice_field("ai_vassal_desire", "replaced with `ai_subject_desire` in 1.16");
111 vd.field_script_value("ai_subject_desire", &mut sc);
112 vd.field_validated_block("liege_modifier", |block, data| {
113 let vd = Validator::new(block, data);
114 validate_modifs(block, data, ModifKinds::Character, vd);
115 });
116 vd.advice_field("vassal_modifier", "replaced with `subject_modifier` in 1.16");
117 vd.field_validated_block("subject_modifier", |block, data| {
118 let vd = Validator::new(block, data);
119 validate_modifs(block, data, ModifKinds::Character, vd);
120 });
121 vd.field_trigger("is_shown", Tooltipped::No, &mut sc);
122 vd.field_trigger("is_valid", Tooltipped::Yes, &mut sc);
123 vd.field_script_value("tax_factor", &mut sc);
124 vd.field_script_value("levies_factor", &mut sc);
125 vd.field_script_value("herd_factor", &mut sc);
126 });
127 });
128 }
129}
130
131impl DbKind for SubjectContractGroup {
132 fn validate(&self, key: &Token, block: &Block, data: &Everything) {
133 data.verify_exists(Item::Localization, key);
134 let loca = format!("{key}_desc");
135 data.verify_exists_implied(Item::Localization, &loca, key);
136
137 let mut vd = Validator::new(block, data);
138 vd.field_list_items("contracts", Item::SubjectContract);
139 vd.field_value("modify_contract_layout");
140 vd.field_bool("is_tributary");
141
142 let mut sc = ScopeContext::new(Scopes::Character, key);
143 sc.define_name("suzerain", Scopes::Character, key);
144
145 if block.get_field_bool("is_tributary").unwrap_or(false) {
146 vd.field_trigger("is_valid_tributary_contract", Tooltipped::Yes, &mut sc);
147 vd.field_trigger("tributary_can_break_free", Tooltipped::Yes, &mut sc);
148 vd.field_item("suzerain_line_type", Item::LineType);
149 vd.field_item("tributary_line_type", Item::LineType);
150 vd.field_bool("should_show_as_suzerain_realm_name");
151 vd.field_bool("should_show_as_suzerain_realm_color");
152 vd.field_bool("tributary_heir_succession");
153 vd.field_bool("suzerain_heir_succession");
154 } else {
155 vd.ban_field("is_valid_tributary_contract", || "is_tributary = yes");
156 vd.ban_field("tributary_can_break_free", || "is_tributary = yes");
157 vd.ban_field("suzerain_line_type", || "is_tributary = yes");
158 vd.ban_field("tributary_line_type", || "is_tributary = yes");
159 vd.ban_field("should_show_as_suzerain_realm_name", || "is_tributary = yes");
160 vd.ban_field("should_show_as_suzerain_realm_color", || "is_tributary = yes");
161 vd.ban_field("tributary_heir_succession", || "is_tributary = yes");
162 vd.ban_field("suzerain_heir_succession", || "is_tributary = yes");
163 }
164 }
165}