tiger_lib/ck3/data/
subjectcontract.rs

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}